pull/287/merge
BenZonneveld 3 years ago committed by GitHub
commit 5337027298
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
  1. 1
      .github/workflows/build.yml
  2. 5
      .gitignore
  3. 2
      Synth_Dexed
  4. 3
      build.sh
  5. 2
      circle-stdlib
  6. 42
      rebuild.sh
  7. 186
      serial.h
  8. 276
      src/mididevice.cpp
  9. 22
      src/mididevice.h
  10. 17
      src/minidexed.cpp
  11. 1
      src/minidexed.h
  12. 6
      src/serialmididevice.cpp

@ -86,4 +86,3 @@ jobs:
set -ex
wget -c https://github.com/probonopd/uploadtool/raw/master/upload.sh
bash ./upload.sh ./MiniDexed*.zip

5
.gitignore vendored

@ -42,6 +42,9 @@ sdcard
*.zip
*.img
# Visual Studio artefacts
.vs
# Editor related files
*.swp
*.swo
*.swo

@ -1 +1 @@
Subproject commit 8c677ceb4b3fb73f8643e30ff6cf4158dc8b9e53
Subproject commit 28ef0da952b5c3b20204f14a56724290235b154d

@ -14,6 +14,7 @@ else
export TOOLCHAIN_PREFIX="arm-none-eabi-"
fi
# Define system options
OPTIONS="-o USE_PWM_AUDIO_ON_ZERO -o SAVE_VFP_REGS_ON_IRQ -o REALTIME -o SCREEN_DMA_BURST_LENGTH=1"
if [ "${RPI}" -gt "1" ]; then
@ -21,6 +22,8 @@ if [ "${RPI}" -gt "1" ]; then
fi
# Build circle-stdlib library
# Apply NL to NL+CR on output fix
cp serial.h circle-stdlib/libs/circle/include/circle/serial.h
cd circle-stdlib/
make mrproper || true
./configure -r ${RPI} --prefix "${TOOLCHAIN_PREFIX}" ${OPTIONS} -o KERNEL_MAX_SIZE=0x400000

@ -1 +1 @@
Subproject commit 61cf3a47bf93628039078b7c840e44432e52343e
Subproject commit 09d328c28f831ffe8910104c21aff32fa1b08fd5

@ -0,0 +1,42 @@
export RPI=3
export CLEAN="true"
export BUILDCIRCLE="true"
export PATH=$(readlink -f ./gcc-*/bin/):$PATH
# Build dependencies and MiniDexed
./build.sh
cp ./src/kernel*.img ./kernels/
if [[ $? -ne 0 ]] ; then
exit
fi
# Get Raspberry Pi boot files
cd ./circle-stdlib/libs/circle/boot
make
if [ "${RPI}" -gt 2 ]
then
make armstub64
fi
cd -
# Make zip that contains Raspberry Pi 4 boot files. The contents can be copied to a FAT32 formatted partition on a microSD card
#cd sdcard
#../getsysex.sh
#cd ..
cp -r ./circle-stdlib/libs/circle/boot/* sdcard
if [[ $? -ne 0 ]] ; then
exit
fi
rm -rf sdcard/config*.txt sdcard/README sdcard/Makefile sdcard/armstub sdcard/COPYING.linux
cp ./src/config.txt ./src/*img sdcard/
if [[ $? -ne 0 ]] ; then
exit
fi
echo "usbspeed=full" > sdcard/cmdline.txt
cd sdcard
cp ../kernels/* . || true
if [[ $? -ne 0 ]] ; then
exit
fi
zip -r ../MiniDexed_$(date +%Y-%m-%d).zip *
cd -

@ -0,0 +1,186 @@
//
/// \file serial.h
//
// Circle - A C++ bare metal environment for Raspberry Pi
// Copyright (C) 2014-2021 R. Stange <rsta2@o2online.de>
//
// This program 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
// (at your option) any later version.
//
// This program 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 this program. If not, see <http://www.gnu.org/licenses/>.
//
#ifndef _circle_serial_h
#define _circle_serial_h
#include <circle/device.h>
#include <circle/interrupt.h>
#include <circle/gpiopin.h>
#include <circle/spinlock.h>
#include <circle/sysconfig.h>
#include <circle/types.h>
/// \class CSerialDevice
/// \brief Driver for PL011 UART
///
/// \details GPIO pin mapping (chip numbers)
/// nDevice | TXD | RXD | Support
/// :-----: | :----: | :----: | :------
/// 0 | GPIO14 | GPIO15 | All boards
/// ^ | GPIO32 | GPIO33 | Compute Modules
/// ^ | GPIO36 | GPIO37 | Compute Modules
/// 1 | | | None (AUX)
/// 2 | GPIO0 | GPIO1 | Raspberry Pi 4 only
/// 3 | GPIO4 | GPIO5 | Raspberry Pi 4 only
/// 4 | GPIO8 | GPIO9 | Raspberry Pi 4 only
/// 5 | GPIO12 | GPIO13 | Raspberry Pi 4 only
/// GPIO32/33 and GPIO36/37 can be selected with system option SERIAL_GPIO_SELECT.\n
/// GPIO0/1 are normally reserved for ID EEPROM.\n
/// Handshake lines CTS and RTS are not supported.
#if RASPPI < 4
#define SERIAL_DEVICES 1
#else
#define SERIAL_DEVICES 6
#endif
#define SERIAL_BUF_SIZE 2048 // must be a power of 2
#define SERIAL_BUF_MASK (SERIAL_BUF_SIZE-1)
// serial options
#define SERIAL_OPTION_ONLCR 0 ///< Translate NL to NL+CR on output (default)
// returned from Read/Write as negative value
#define SERIAL_ERROR_BREAK 1
#define SERIAL_ERROR_OVERRUN 2
#define SERIAL_ERROR_FRAMING 3
#define SERIAL_ERROR_PARITY 4
class CSerialDevice : public CDevice
{
public:
enum TParity
{
ParityNone,
ParityOdd,
ParityEven,
ParityUnknown
};
public:
#ifndef USE_RPI_STUB_AT
/// \param pInterruptSystem Pointer to interrupt system object (or 0 for polling driver)
/// \param bUseFIQ Use FIQ instead of IRQ
/// \param nDevice Device number (see: GPIO pin mapping)
CSerialDevice (CInterruptSystem *pInterruptSystem = 0, boolean bUseFIQ = FALSE,
unsigned nDevice = 0);
~CSerialDevice (void);
#endif
/// \param nBaudrate Baud rate in bits per second
/// \param nDataBits Number of data bits (5..8, default 8)
/// \param nStopBits Number of stop bits (1..2, default 1)
/// \param Parity Parity setting (ParityNone (default), ParityOdd or ParityEven)
/// \return Operation successful?
#ifndef USE_RPI_STUB_AT
boolean Initialize (unsigned nBaudrate = 115200,
unsigned nDataBits = 8, unsigned nStopBits = 1,
TParity Parity = ParityNone);
#else
boolean Initialize (unsigned nBaudrate = 115200);
#endif
/// \param pBuffer Pointer to data to be sent
/// \param nCount Number of bytes to be sent
/// \return Number of bytes successfully sent (< 0 on error)
int Write (const void *pBuffer, size_t nCount);
#ifndef USE_RPI_STUB_AT
/// \param pBuffer Pointer to buffer for received data
/// \param nCount Maximum number of bytes to be received
/// \return Number of bytes received (0 no data available, < 0 on error)
int Read (void *pBuffer, size_t nCount);
/// \return Serial options mask (see serial options)
unsigned GetOptions (void) const;
/// \param nOptions Serial options mask (see serial options)
void SetOptions (unsigned nOptions);
typedef void TMagicReceivedHandler (void);
/// \param pMagic String for which is searched in the received data\n
/// (must remain valid after return from this method)
/// \param pHandler Handler which is called, when the magic string is found
/// \note Does only work with interrupt driver.
void RegisterMagicReceivedHandler (const char *pMagic, TMagicReceivedHandler *pHandler);
protected:
/// \return Number of bytes buffer space available for Write()
/// \note Does only work with interrupt driver.
unsigned AvailableForWrite (void);
/// \return Number of bytes already received available for Read()
/// \note Does only work with interrupt driver.
unsigned AvailableForRead (void);
/// \return Next received byte which will be returned by Read() (-1 if no data available)
/// \note Does only work with interrupt driver.
int Peek (void);
/// \brief Waits until all written bytes have been sent out
void Flush (void);
private:
boolean Write (u8 uchChar);
void InterruptHandler (void);
static void InterruptStub (void *pParam);
private:
CInterruptSystem *m_pInterruptSystem;
boolean m_bUseFIQ;
unsigned m_nDevice;
uintptr m_nBaseAddress;
boolean m_bValid;
#if SERIAL_GPIO_SELECT == 14
CGPIOPin m_GPIO32;
CGPIOPin m_GPIO33;
#endif
CGPIOPin m_TxDPin;
CGPIOPin m_RxDPin;
u8 m_RxBuffer[SERIAL_BUF_SIZE];
volatile unsigned m_nRxInPtr;
volatile unsigned m_nRxOutPtr;
volatile int m_nRxStatus;
u8 m_TxBuffer[SERIAL_BUF_SIZE];
volatile unsigned m_nTxInPtr;
volatile unsigned m_nTxOutPtr;
unsigned m_nOptions;
const char *m_pMagic;
const char *m_pMagicPtr;
TMagicReceivedHandler *m_pMagicReceivedHandler;
CSpinLock m_SpinLock;
CSpinLock m_LineSpinLock;
static unsigned s_nInterruptUseCount;
static CInterruptSystem *s_pInterruptSystem;
static boolean s_bUseFIQ;
static volatile u32 s_nInterruptDeviceMask;
static CSerialDevice *s_pThis[SERIAL_DEVICES];
#endif
};
#endif

@ -34,18 +34,6 @@ LOGMODULE ("mididevice");
#define MIDI_NOTE_ON 0b1001
#define MIDI_AFTERTOUCH 0b1010 // TODO
#define MIDI_CONTROL_CHANGE 0b1011
#define MIDI_CC_BANK_SELECT_MSB 0 // TODO
#define MIDI_CC_MODULATION 1
#define MIDI_CC_VOLUME 7
#define MIDI_CC_PAN_POSITION 10
#define MIDI_CC_BANK_SELECT_LSB 32
#define MIDI_CC_BANK_SUSTAIN 64
#define MIDI_CC_RESONANCE 71
#define MIDI_CC_FREQUENCY_CUTOFF 74
#define MIDI_CC_REVERB_LEVEL 91
#define MIDI_CC_DETUNE_LEVEL 94
#define MIDI_CC_ALL_SOUND_OFF 120
#define MIDI_CC_ALL_NOTES_OFF 123
#define MIDI_PROGRAM_CHANGE 0b1100
#define MIDI_PITCH_BEND 0b1110
@ -184,10 +172,11 @@ void CMIDIDevice::MIDIMessageHandler (const u8 *pMessage, size_t nLength, unsign
{
// MIDI SYSEX per MIDI channel
uint8_t ucSysExChannel = (pMessage[2] & 0x07);
if (m_ChannelMap[nTG] == ucSysExChannel || m_ChannelMap[nTG] == OmniMode)
if ( 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);
//printf("MIDI-SYSEX: channel: %u, len: %lu, TG: %u",m_ChannelMap[nTG],nLength,nTG);
HandleSystemExclusive(pMessage, nLength, nCable, ucSysExChannel);
}
}
else
@ -267,7 +256,7 @@ void CMIDIDevice::MIDIMessageHandler (const u8 *pMessage, size_t nLength, unsign
m_pSynthesizer->SetReverbSend (maplong (pMessage[2], 0, 127, 0, 99), nTG);
break;
case MIDI_CC_DETUNE_LEVEL:
case MIDI_CC_DETUNE_LEVEL+32:
if (pMessage[2] == 0)
{
// "0 to 127, with 0 being no celeste (detune) effect applied at all."
@ -333,7 +322,20 @@ void CMIDIDevice::HandleSystemExclusive(const uint8_t* pMessage, const size_t nL
{
int16_t sysex_return;
if ( nTG >= CConfig::ToneGenerators ) return;
sysex_return = m_pSynthesizer->checkSystemExclusive(pMessage, nLength, nTG);
uint8_t instanceID = pMessage[2]&0xF;
if ( sysex_return == -11 && pMessage[0] == 0xF0 && pMessage[1] == 0x43 && nLength == 4 )
{
if ((pMessage[2] & 0x70) == 0x30) // Send config request
sysex_return = 600;
if ((pMessage[2] & 0x70) == 0x40) // Send Bank Name request
sysex_return = 601;
}
if ( instanceID != nTG ) { printf("WARNING instanceID and nTG do not match!!!!!\n"); }
LOGDBG("SYSEX handler return value: %d", sysex_return);
switch (sysex_return)
@ -372,60 +374,152 @@ void CMIDIDevice::HandleSystemExclusive(const uint8_t* pMessage, const size_t nL
break;
case 64:
LOGDBG("SysEx Function parameter change: %d Value %d",pMessage[4],pMessage[5]);
m_pSynthesizer->setMonoMode(pMessage[5],nTG);
m_pSynthesizer->setMonoMode(pMessage[5], instanceID);
break;
case 65:
LOGDBG("SysEx Function parameter change: %d Value %d",pMessage[4],pMessage[5]);
m_pSynthesizer->setPitchbendRange(pMessage[5],nTG);
m_pSynthesizer->setPitchbendRange(pMessage[5],instanceID);
break;
case 66:
LOGDBG("SysEx Function parameter change: %d Value %d",pMessage[4],pMessage[5]);
m_pSynthesizer->setPitchbendStep(pMessage[5],nTG);
m_pSynthesizer->setPitchbendStep(pMessage[5],instanceID);
break;
case 67:
LOGDBG("SysEx Function parameter change: %d Value %d",pMessage[4],pMessage[5]);
m_pSynthesizer->setPortamentoMode(pMessage[5],nTG);
m_pSynthesizer->setPortamentoMode(pMessage[5],instanceID);
break;
case 68:
LOGDBG("SysEx Function parameter change: %d Value %d",pMessage[4],pMessage[5]);
m_pSynthesizer->setPortamentoGlissando(pMessage[5],nTG);
m_pSynthesizer->setPortamentoGlissando(pMessage[5],instanceID);
break;
case 69:
LOGDBG("SysEx Function parameter change: %d Value %d",pMessage[4],pMessage[5]);
m_pSynthesizer->setPortamentoTime(pMessage[5],nTG);
m_pSynthesizer->setPortamentoTime(pMessage[5],instanceID);
break;
case 70:
LOGDBG("SysEx Function parameter change: %d Value %d",pMessage[4],pMessage[5]);
m_pSynthesizer->setModWheelRange(pMessage[5],nTG);
m_pSynthesizer->setModWheelRange(pMessage[5],instanceID);
break;
case 71:
LOGDBG("SysEx Function parameter change: %d Value %d",pMessage[4],pMessage[5]);
m_pSynthesizer->setModWheelTarget(pMessage[5],nTG);
m_pSynthesizer->setModWheelTarget(pMessage[5],instanceID);
break;
case 72:
LOGDBG("SysEx Function parameter change: %d Value %d",pMessage[4],pMessage[5]);
m_pSynthesizer->setFootControllerRange(pMessage[5],nTG);
m_pSynthesizer->setFootControllerRange(pMessage[5],instanceID);
break;
case 73:
LOGDBG("SysEx Function parameter change: %d Value %d",pMessage[4],pMessage[5]);
m_pSynthesizer->setFootControllerTarget(pMessage[5],nTG);
m_pSynthesizer->setFootControllerTarget(pMessage[5],instanceID);
break;
case 74:
LOGDBG("SysEx Function parameter change: %d Value %d",pMessage[4],pMessage[5]);
m_pSynthesizer->setBreathControllerRange(pMessage[5],nTG);
m_pSynthesizer->setBreathControllerRange(pMessage[5],instanceID);
break;
case 75:
LOGDBG("SysEx Function parameter change: %d Value %d",pMessage[4],pMessage[5]);
m_pSynthesizer->setBreathControllerTarget(pMessage[5],nTG);
m_pSynthesizer->setBreathControllerTarget(pMessage[5],instanceID);
break;
case 76:
LOGDBG("SysEx Function parameter change: %d Value %d",pMessage[4],pMessage[5]);
m_pSynthesizer->setAftertouchRange(pMessage[5],nTG);
m_pSynthesizer->setAftertouchRange(pMessage[5],instanceID);
break;
case 77:
LOGDBG("SysEx Function parameter change: %d Value %d",pMessage[4],pMessage[5]);
m_pSynthesizer->setAftertouchTarget(pMessage[5],nTG);
m_pSynthesizer->setAftertouchTarget(pMessage[5],instanceID);
break;
/* BeZo patches */
case 78: // bank select
LOGDBG("Bank Select for TG %i\n", instanceID);
m_pSynthesizer->BankSelectLSB (pMessage[5], instanceID);
break;
case 79: // pgm select
LOGDBG("Patch Select for TG %i\n", instanceID);
m_pSynthesizer->ProgramChange (pMessage[5], instanceID);
break;
case 80: // Set midi channel
LOGDBG("Set midi channel for TG %i", instanceID);
m_pSynthesizer->SetMIDIChannel(pMessage[5], instanceID);
break;
case 81: // Set Cutoff
LOGDBG("Set Cutoff for TG %i", instanceID);
m_pSynthesizer->SetCutoff(pMessage[5], instanceID);
break;
case 82: // Set Reso
LOGDBG("Set Resonanece for TG %i", instanceID);
m_pSynthesizer->SetResonance(pMessage[5], instanceID);
break;
case 83: // Reverb level
LOGDBG("Set Reverb Level for TG %i", instanceID);
m_pSynthesizer->SetReverbSend (pMessage[5], instanceID);
break;
case 84: // Transpose
LOGDBG("Set Transpose for TG %i", instanceID);
// m_pSynthesizer->SetTranspose (pMessage[5], instanceID);
break;
case 85: // Detune
LOGDBG("Set detune for TG %i", instanceID);
if (pMessage[5] == 0)
{
// "0 to 127, with 0 being no celeste (detune) effect applied at all."
m_pSynthesizer->SetMasterTune (0, instanceID);
}
else
{
m_pSynthesizer->SetMasterTune (maplong (pMessage[5], 1, 127, -99, 99), instanceID);
}
break;
case 86: // Panning
LOGDBG("Set panning for TG %i", instanceID);
m_pSynthesizer->SetPan(pMessage[5], instanceID);
break;
case 87: // Note Limit Low
LOGDBG("Set Note Limit High mode for TG %i", instanceID);
break;
case 88: // Note Limit High
LOGDBG("Set Note Limit High mode for TG %i", instanceID);
break;
case 89: // Compressor toggle
LOGDBG("Set Compressor ");
m_pSynthesizer->SetParameter (CMiniDexed::ParameterCompressorEnable, pMessage[5] );
break;
case 90: // Reverb toggle
LOGDBG("Set Reverb Enable");
m_pSynthesizer->SetParameter (CMiniDexed::ParameterReverbEnable, pMessage[5] );
break;
case 91: // Reverb Size
LOGDBG("Set Reverb Size");
m_pSynthesizer->SetParameter (CMiniDexed::ParameterReverbSize, pMessage[5] );
break;
case 92: // Reverb Low Damp
LOGDBG("Set Reverb Low Damp");
m_pSynthesizer->SetParameter (CMiniDexed::ParameterReverbLowDamp, pMessage[5]);
break;
case 93: // Reverb High Damp
LOGDBG("Set Reverb High Damp");
m_pSynthesizer->SetParameter (CMiniDexed::ParameterReverbHighDamp, pMessage[5] );
break;
case 94: // Reverb Lowpass
LOGDBG("Set Reverb Low pass");
m_pSynthesizer->SetParameter (CMiniDexed::ParameterReverbLowPass, pMessage[5]);
break;
case 95: // Reverb Diffusion
LOGDBG("Set Reverb Diffusion");
m_pSynthesizer->SetParameter (CMiniDexed::ParameterReverbDiffusion, pMessage[5] );
break;
case 96: // Reverb Master Level
LOGDBG("Set Reverb Master Level");
m_pSynthesizer->SetParameter (CMiniDexed::ParameterReverbLevel, pMessage[5] );
break;
case 600: // Config requestnTG
LOGDBG("Config request received\n");
SendSystemExclusiveConfig();
break;
case 601:
LOGDBG("Get Bank Name request received\n");
SendBankName(instanceID);
break;
/* End of BeZo patches */
case 100:
// load sysex-data into voice memory
LOGDBG("One Voice bulk upload");
@ -440,7 +534,7 @@ void CMIDIDevice::HandleSystemExclusive(const uint8_t* pMessage, const size_t nL
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);
m_pSynthesizer->setVoiceDataElement(pMessage[4] + ((pMessage[3] & 0x03) * 128), pMessage[5],instanceID);
switch(pMessage[4] + ((pMessage[3] & 0x03) * 128))
{
case 134:
@ -451,13 +545,13 @@ void CMIDIDevice::HandleSystemExclusive(const uint8_t* pMessage, const size_t nL
else if(sysex_return >= 500 && sysex_return < 600)
{
LOGDBG("SysEx send voice %u request",sysex_return-500);
SendSystemExclusiveVoice(sysex_return-500, nCable, nTG);
SendSystemExclusiveVoice(sysex_return-500, instanceID);
}
break;
}
}
void CMIDIDevice::SendSystemExclusiveVoice(uint8_t nVoice, const unsigned nCable, uint8_t nTG)
void CMIDIDevice::SendSystemExclusiveVoice(uint8_t nVoice, uint8_t nTG)
{
uint8_t voicedump[163];
@ -473,3 +567,121 @@ void CMIDIDevice::SendSystemExclusiveVoice(uint8_t nVoice, const unsigned nCable
// LOGDBG("Send SYSEX voice dump %u to \"%s\"",nVoice,Iterator->first.c_str());
}
}
void CMIDIDevice::SendSystemExclusiveConfig()
{
uint8_t count = 0;
uint8_t configdump[204];
configdump[count++] = 0xF0;
configdump[count++] = 0x43;
configdump[count++] = 0x31;
// FX Settings
configdump[count++] = ((m_pSynthesizer->GetParameter(CMiniDexed::ParameterCompressorEnable) & 0x7F)<<1) |
m_pSynthesizer->GetParameter(CMiniDexed::ParameterReverbEnable) & 0x7F;
configdump[count++] = m_pSynthesizer->GetParameter(CMiniDexed::ParameterReverbSize) & 0x7F;
configdump[count++] = m_pSynthesizer->GetParameter(CMiniDexed::ParameterReverbHighDamp) & 0x7F;
configdump[count++] = m_pSynthesizer->GetParameter(CMiniDexed::ParameterReverbLowDamp) & 0x7F;
configdump[count++] = m_pSynthesizer->GetParameter(CMiniDexed::ParameterReverbLowPass) & 0x7F;
configdump[count++] = m_pSynthesizer->GetParameter(CMiniDexed::ParameterReverbDiffusion) & 0x7F;
configdump[count++] = m_pSynthesizer->GetParameter(CMiniDexed::ParameterReverbLevel) & 0x7F;
configdump[count++] = m_pSynthesizer->getMasterVolume() & 0x7F;
for ( uint8_t instance = 0 ; instance < CConfig::ToneGenerators; instance++)
{
configdump[count++] = m_pSynthesizer->GetTGParameter(CMiniDexed::TGParameterVoiceBank, instance) & 0x7F;
configdump[count++] = m_pSynthesizer->GetTGParameter(CMiniDexed::TGParameterProgram, instance) & 0x7F;
configdump[count++] = m_pSynthesizer->GetTGParameter(CMiniDexed::TGParameterMIDIChannel, instance) & 0x7F;
configdump[count++] = m_pSynthesizer->GetTGParameter(CMiniDexed::TGParameterVolume, instance) & 0x7F;
configdump[count++] = m_pSynthesizer->GetTGParameter(CMiniDexed::TGParameterPan, instance) & 0x7F;
int16_t mastertune = m_pSynthesizer->GetTGParameter(CMiniDexed::TGParameterMasterTune, instance);
configdump[count++] = (mastertune >> 9)&0x7f;
configdump[count++] = (mastertune & 0x7f);
configdump[count++] = m_pSynthesizer->GetTGParameter(CMiniDexed::TGParameterCutoff, instance) & 0x7F;
configdump[count++] = m_pSynthesizer->GetTGParameter(CMiniDexed::TGParameterResonance, instance) & 0x7F;
configdump[count++] = 0; // Note limit low
configdump[count++] = 127; // Note limit high
configdump[count++] = 0; // Note shift
configdump[count++] = m_pSynthesizer->GetTGParameter(CMiniDexed::TGParameterReverbSend, instance) & 0x7F;
configdump[count++] = m_pSynthesizer->GetTGParameter(CMiniDexed::TGParameterPitchBendRange, instance) & 0x7F;
configdump[count++] = m_pSynthesizer->GetTGParameter(CMiniDexed::TGParameterPitchBendStep, instance) & 0x7F;
configdump[count++] = m_pSynthesizer->GetTGParameter(CMiniDexed::TGParameterPortamentoMode, instance) & 0x7F;
configdump[count++] = m_pSynthesizer->GetTGParameter(CMiniDexed::TGParameterPortamentoGlissando, instance) & 0x7F;
configdump[count++] = m_pSynthesizer->GetTGParameter(CMiniDexed::TGParameterPortamentoTime, instance) & 0x7F;
configdump[count++] = 0;
configdump[count++] = 0;
configdump[count++] = 0;
configdump[count++] = 0;
configdump[count++] = 0;
configdump[count++] = 0;
}
configdump[count++] = 0xF7;
TDeviceMap::const_iterator Iterator;
// send voice dump to all MIDI interfaces
for(Iterator = s_DeviceMap.begin(); Iterator != s_DeviceMap.end(); ++Iterator)
{
Iterator->second->Send (configdump, count);
LOGDBG("Send SYSEX config dump to \"%s\"",Iterator->first.c_str());
}
}
void CMIDIDevice::SendProgramChange(uint8_t pgm, uint8_t nTG)
{
uint8_t PgmChange[2] = { (uint8_t)(0xC0|(nTG & 0x0F)), (uint8_t)(pgm & 0x7f) };
TDeviceMap::const_iterator Iterator;
// send voice dump to all MIDI interfaces
for(Iterator = s_DeviceMap.begin(); Iterator != s_DeviceMap.end(); ++Iterator)
{
Iterator->second->Send (PgmChange, 2);
LOGDBG("Send Program Change %i to \"%s\"",pgm&0x7f,Iterator->first.c_str());
}
}
void CMIDIDevice::SendBankChange(uint8_t bank, uint8_t nTG)
{
SendCtrlChange14Bit(0, bank, nTG);
}
void CMIDIDevice::SendCtrlChange(uint8_t ctrl, uint8_t val, uint8_t nTG)
{
uint8_t CtrlMsg[3] = { (uint8_t)(0xB0|(nTG & 0x0F)), (uint8_t)(ctrl&0x7f), (uint8_t)(val&0x7f) };
TDeviceMap::const_iterator Iterator;
// send voice dump to all MIDI interfaces
for(Iterator = s_DeviceMap.begin(); Iterator != s_DeviceMap.end(); ++Iterator)
{
Iterator->second->Send (CtrlMsg, 3);
LOGDBG("Send Ctrl change %02X = %i to \"%s\"",ctrl&0x7f, val&0x7f,Iterator->first.c_str());
}
}
void CMIDIDevice::SendCtrlChange14Bit(uint8_t ctrl, int16_t val, uint8_t nTG)
{
uint8_t lsb = (val & 0x7f);
uint8_t msb = (val >> 9)&0x7f;
SendCtrlChange(ctrl,msb,nTG);
SendCtrlChange(ctrl+32, lsb, nTG);
}
void CMIDIDevice::SendBankName(uint8_t nTG)
{
char *bankname = (char*)calloc(32,sizeof(char));
snprintf(bankname,sizeof(bankname), "%s", m_pSynthesizer->GetSysExFileLoader()->GetBankName(m_pSynthesizer->GetTGParameter(CMiniDexed::TGParameterVoiceBank,nTG)).c_str());
uint8_t banksysex[40] = { 0xF0, 0x43, (uint8_t)(0x50|nTG), 0,0,32 };
memcpy(banksysex+6,bankname,32);
banksysex[38] = 00;
banksysex[39] = 0xF7;
TDeviceMap::const_iterator Iterator;
// send voice dump to all MIDI interfaces
for(Iterator = s_DeviceMap.begin(); Iterator != s_DeviceMap.end(); ++Iterator)
{
Iterator->second->Send (banksysex, sizeof(banksysex)*sizeof(uint8_t));
LOGDBG("Send Bank Name Sysex to \"%s\"",Iterator->first.c_str());
}
}

@ -29,6 +29,19 @@
#include <circle/types.h>
#include <circle/spinlock.h>
#define MIDI_CC_BANK_SELECT_MSB 0 // TODO
#define MIDI_CC_MODULATION 1
#define MIDI_CC_VOLUME 7
#define MIDI_CC_PAN_POSITION 10
#define MIDI_CC_BANK_SELECT_LSB 32
#define MIDI_CC_BANK_SUSTAIN 64
#define MIDI_CC_RESONANCE 71
#define MIDI_CC_FREQUENCY_CUTOFF 74
#define MIDI_CC_REVERB_LEVEL 91
#define MIDI_CC_DETUNE_LEVEL 94
#define MIDI_CC_ALL_SOUND_OFF 120
#define MIDI_CC_ALL_NOTES_OFF 123
class CMiniDexed;
class CMIDIDevice
@ -50,8 +63,13 @@ public:
u8 GetChannel (unsigned nTG) const;
virtual void Send (const u8 *pMessage, size_t nLength, unsigned nCable = 0) {}
virtual void SendSystemExclusiveVoice(uint8_t nVoice, const unsigned nCable, uint8_t nTG);
virtual void SendSystemExclusiveVoice(uint8_t nVoice, uint8_t nTG);
virtual void SendSystemExclusiveConfig();
virtual void SendProgramChange(uint8_t pgm, uint8_t nTG);
virtual void SendBankChange(uint8_t bank, uint8_t nTG);
virtual void SendBankName( uint8_t nTG);
virtual void SendCtrlChange(uint8_t ctrl, uint8_t val, uint8_t nTG);
virtual void SendCtrlChange14Bit(uint8_t ctrl, int16_t val, uint8_t nTG);
protected:
void MIDIMessageHandler (const u8 *pMessage, size_t nLength, unsigned nCable = 0);
void AddDevice (const char *pDeviceName);

@ -346,6 +346,7 @@ void CMiniDexed::BankSelectLSB (unsigned nBankLSB, unsigned nTG)
assert (nTG < CConfig::ToneGenerators);
m_nVoiceBankID[nTG] = nBankLSB;
m_SerialMIDI.SendBankChange(m_nVoiceBankID[nTG],nTG);
m_UI.ParameterChanged ();
}
@ -362,7 +363,7 @@ void CMiniDexed::ProgramChange (unsigned nProgram, unsigned nTG)
assert (m_pTG[nTG]);
m_pTG[nTG]->loadVoiceParameters (Buffer);
m_SerialMIDI.SendSystemExclusiveVoice(nProgram,0,nTG);
m_SerialMIDI.SendProgramChange(m_nProgram[nTG],nTG);
m_UI.ParameterChanged ();
}
@ -376,7 +377,7 @@ void CMiniDexed::SetVolume (unsigned nVolume, unsigned nTG)
assert (m_pTG[nTG]);
m_pTG[nTG]->setGain (nVolume / 127.0f);
m_SerialMIDI.SendCtrlChange(MIDI_CC_VOLUME, m_nVolume[nTG], nTG);
m_UI.ParameterChanged ();
}
@ -390,6 +391,7 @@ void CMiniDexed::SetPan (unsigned nPan, unsigned nTG)
tg_mixer->pan(nTG,mapfloat(nPan,0,127,0.0f,1.0f));
reverb_send_mixer->pan(nTG,mapfloat(nPan,0,127,0.0f,1.0f));
m_SerialMIDI.SendCtrlChange(MIDI_CC_PAN_POSITION, m_nPan[nTG], nTG);
m_UI.ParameterChanged ();
}
@ -402,6 +404,8 @@ void CMiniDexed::SetReverbSend (unsigned nReverbSend, unsigned nTG)
reverb_send_mixer->gain(nTG,mapfloat(nReverbSend,0,99,0.0f,1.0f));
m_SerialMIDI.SendCtrlChange(MIDI_CC_REVERB_LEVEL, m_nReverbSend[nTG], nTG);
m_UI.ParameterChanged ();
}
@ -415,6 +419,8 @@ void CMiniDexed::SetMasterTune (int nMasterTune, unsigned nTG)
assert (m_pTG[nTG]);
m_pTG[nTG]->setMasterTune ((int8_t) nMasterTune);
m_SerialMIDI.SendCtrlChange14Bit(MIDI_CC_DETUNE_LEVEL, m_nMasterTune[nTG], nTG);
m_UI.ParameterChanged ();
}
@ -428,6 +434,8 @@ void CMiniDexed::SetCutoff (int nCutoff, unsigned nTG)
assert (m_pTG[nTG]);
m_pTG[nTG]->setFilterCutoff (mapfloat (nCutoff, 0, 99, 0.0f, 1.0f));
m_SerialMIDI.SendCtrlChange(MIDI_CC_FREQUENCY_CUTOFF, m_nCutoff[nTG], nTG);
m_UI.ParameterChanged ();
}
@ -441,6 +449,8 @@ void CMiniDexed::SetResonance (int nResonance, unsigned nTG)
assert (m_pTG[nTG]);
m_pTG[nTG]->setFilterResonance (mapfloat (nResonance, 0, 99, 0.0f, 1.0f));
m_SerialMIDI.SendCtrlChange(MIDI_CC_RESONANCE, m_nResonance[nTG], nTG);
m_UI.ParameterChanged ();
}
@ -1194,7 +1204,8 @@ void CMiniDexed::getSysExVoiceDump(uint8_t* dest, uint8_t nTG)
dest[0] = 0xF0; // SysEx start
dest[1] = 0x43; // ID=Yamaha
dest[2] = GetTGParameter(TGParameterMIDIChannel, nTG); // Sub-status and MIDI channel
// dest[2] = GetTGParameter(TGParameterMIDIChannel, nTG); // Sub-status and MIDI channel
dest[2] = nTG; // Sub-status and MIDI channel/Instance ID
dest[3] = 0x00; // Format number (0=1 voice)
dest[4] = 0x01; // Byte count MSB
dest[5] = 0x1B; // Byte count LSB

@ -166,6 +166,7 @@ public:
bool DoSavePerformance (void);
void setMasterVolume (float32_t vol);
uint8_t getMasterVolume() { return (127*nMasterVolume); }
private:
int16_t ApplyNoteLimits (int16_t pitch, unsigned nTG); // returns < 0 to ignore note

@ -161,5 +161,11 @@ void CSerialMIDIDevice::Process (void)
void CSerialMIDIDevice::Send (const u8 *pMessage, size_t nLength, unsigned nCable)
{
// for (size_t i= 0; i < nLength; i++)
// {
// printf("%02X, ",pMessage[i]);
// if ( (i+1)%16 == 0 ) printf("\n");
// }
// printf("\n");
m_SendBuffer.Write (pMessage, nLength);
}

Loading…
Cancel
Save