From 4755c5d861d837c68866f98efa46a36fa1dc28a3 Mon Sep 17 00:00:00 2001 From: probonopd Date: Sat, 27 Jan 2024 17:52:22 +0000 Subject: [PATCH 01/38] Performance file handling (#581) Implements #580 * Initial update in performance file handling. This change makes the 6-digit number in the filename indicate a performance "voice number" in MiniDexed. The external filename numbers will now match any Program Change messages using the common MIDI concept of user selecting 1..128 whilst internally they are treated as 0..127. Note: in the case of performances, performance 1 (index 0) is the Default "performance.ini" file for backwards compatibility. Also note that in this version, new performances, when saved, cannot occupy free slots between other performances - they are added to the end. Even though the filename standard gives 6 digit numbers, the actual number of performances is still limited to 256. * Start of subdirectory implementation for performance banks. * Initial version with performance banks, selectable over MIDI only. * Initial implementation of performance bank switching in the UI menu. * Remove debug information, fix few bugs, including PgmUpDown handling and performance numbers out of range. * Bugfixes for legacy cases when no performance directory exists plus some extra checks for saving and deleting performances. * Remove verbose debug options (doh!) * Fix a minor off-by-one error found in review. * Bugfix - removed redundant legacy check that results in out of order performance files being skipped on load. * Fix bug in MIDI button handling commands. * Fix for issue where wrong performance is selected [L] on new save. * Suggested update to UI to show bank/performance numbers. * Make performance bank select asynchronous to MIDI and UI to stop corruptions on loading performances. * Fix an assert that should be a run-time test. * Ensure bank selection works when PCCH is not enabled, and that UI remains consistent when changing banks. --------- Co-authored-by: Kevin <68612569+diyelectromusic@users.noreply.github.com> --- src/mididevice.cpp | 28 +- src/minidexed.cpp | 169 ++++++++++-- src/minidexed.h | 21 ++ src/performanceconfig.cpp | 533 ++++++++++++++++++++++++++++++++------ src/performanceconfig.h | 33 ++- src/uimenu.cpp | 186 +++++++++++-- src/uimenu.h | 2 + src/userinterface.cpp | 3 + 8 files changed, 842 insertions(+), 133 deletions(-) diff --git a/src/mididevice.cpp b/src/mididevice.cpp index 5f231d6..81314ba 100644 --- a/src/mididevice.cpp +++ b/src/mididevice.cpp @@ -184,9 +184,35 @@ void CMIDIDevice::MIDIMessageHandler (const u8 *pMessage, size_t nLength, unsign else { // Perform any MiniDexed level MIDI handling before specific Tone Generators + unsigned nPerfCh = m_pSynthesizer->GetPerformanceSelectChannel(); switch (ucType) { case MIDI_CONTROL_CHANGE: + // Check for performance PC messages + if (nPerfCh != Disabled) + { + if ((ucChannel == nPerfCh) || (nPerfCh == OmniMode)) + { + if (pMessage[1] == MIDI_CC_BANK_SELECT_MSB) + { + m_pSynthesizer->BankSelectMSBPerformance (pMessage[2]); + } + else if (pMessage[1] == MIDI_CC_BANK_SELECT_LSB) + { + m_pSynthesizer->BankSelectLSBPerformance (pMessage[2]); + } + else + { + // Ignore any other CC messages at this time + } + } + } + if (nLength == 3) + { + m_pUI->UIMIDICmdHandler (ucChannel, ucStatus & 0xF0, pMessage[1], pMessage[2]); + } + break; + case MIDI_NOTE_OFF: case MIDI_NOTE_ON: if (nLength < 3) @@ -195,11 +221,11 @@ void CMIDIDevice::MIDIMessageHandler (const u8 *pMessage, size_t nLength, unsign } m_pUI->UIMIDICmdHandler (ucChannel, ucStatus & 0xF0, pMessage[1], pMessage[2]); break; + case MIDI_PROGRAM_CHANGE: // Check for performance PC messages if( m_pConfig->GetMIDIRXProgramChange() ) { - unsigned nPerfCh = m_pSynthesizer->GetPerformanceSelectChannel(); if( nPerfCh != Disabled) { if ((ucChannel == nPerfCh) || (nPerfCh == OmniMode)) diff --git a/src/minidexed.cpp b/src/minidexed.cpp index 71b7247..968320a 100644 --- a/src/minidexed.cpp +++ b/src/minidexed.cpp @@ -53,8 +53,11 @@ CMiniDexed::CMiniDexed (CConfig *pConfig, CInterruptSystem *pInterrupt, m_bSavePerformance (false), m_bSavePerformanceNewFile (false), m_bSetNewPerformance (false), + m_bSetNewPerformanceBank (false), + m_bSetFirstPerformance (false), m_bDeletePerformance (false), - m_bLoadPerformanceBusy(false) + m_bLoadPerformanceBusy(false), + m_bLoadPerformanceBankBusy(false) { assert (m_pConfig); @@ -170,6 +173,8 @@ CMiniDexed::CMiniDexed (CConfig *pConfig, CInterruptSystem *pInterrupt, SetParameter (ParameterCompressorEnable, 1); SetPerformanceSelectChannel(m_pConfig->GetPerformanceSelectChannel()); + + SetParameter (ParameterPerformanceBank, 0); }; bool CMiniDexed::Initialize (void) @@ -227,6 +232,7 @@ bool CMiniDexed::Initialize (void) reverb_send_mixer->gain(i,mapfloat(m_nReverbSend[i],0,99,0.0f,1.0f)); } + m_PerformanceConfig.Init(); if (m_PerformanceConfig.Load ()) { LoadPerformanceParameters(); @@ -236,12 +242,6 @@ bool CMiniDexed::Initialize (void) SetMIDIChannel (CMIDIDevice::OmniMode, 0); } - // load performances file list, and attempt to create the performance folder - if (!m_PerformanceConfig.ListPerformances()) - { - LOGERR ("Cannot create internal Performance folder, new performances can't be created"); - } - // setup and start the sound device if (!m_pSoundDevice->AllocateQueueFrames (m_pConfig->GetChunkSize ())) { @@ -305,14 +305,30 @@ void CMiniDexed::Process (bool bPlugAndPlayUpdated) m_bSavePerformanceNewFile = false; } - if (m_bSetNewPerformance && !m_bLoadPerformanceBusy) + if (m_bSetNewPerformanceBank && !m_bLoadPerformanceBusy && !m_bLoadPerformanceBankBusy) + { + DoSetNewPerformanceBank (); + if (m_nSetNewPerformanceBankID == GetActualPerformanceBankID()) + { + m_bSetNewPerformanceBank = false; + } + + // If there is no pending SetNewPerformance already, then see if we need to find the first performance to load + // NB: If called from the UI, then there will not be a SetNewPerformance, so load the first existing one. + // If called from MIDI, there will probably be a SetNewPerformance alongside the Bank select. + if (!m_bSetNewPerformance && m_bSetFirstPerformance) + { + DoSetFirstPerformance(); + } + } + + if (m_bSetNewPerformance && !m_bSetNewPerformanceBank && !m_bLoadPerformanceBusy && !m_bLoadPerformanceBankBusy) { DoSetNewPerformance (); if (m_nSetNewPerformanceID == GetActualPerformanceID()) { m_bSetNewPerformance = false; } - } if(m_bDeletePerformance) @@ -392,6 +408,11 @@ CSysExFileLoader *CMiniDexed::GetSysExFileLoader (void) return &m_SysExFileLoader; } +CPerformanceConfig *CMiniDexed::GetPerformanceConfig (void) +{ + return &m_PerformanceConfig; +} + void CMiniDexed::BankSelect (unsigned nBank, unsigned nTG) { nBank=constrain((int)nBank,0,16383); @@ -407,6 +428,20 @@ void CMiniDexed::BankSelect (unsigned nBank, unsigned nTG) } } +void CMiniDexed::BankSelectPerformance (unsigned nBank) +{ + nBank=constrain((int)nBank,0,16383); + + if (GetPerformanceConfig ()->IsValidPerformanceBank(nBank)) + { + // Only change if we have the bank loaded + m_nVoiceBankIDPerformance = nBank; + SetNewPerformanceBank (nBank); + + m_UI.ParameterChanged (); + } +} + void CMiniDexed::BankSelectMSB (unsigned nBankMSB, unsigned nTG) { nBankMSB=constrain((int)nBankMSB,0,127); @@ -422,6 +457,12 @@ void CMiniDexed::BankSelectMSB (unsigned nBankMSB, unsigned nTG) m_nVoiceBankIDMSB[nTG] = nBankMSB; } +void CMiniDexed::BankSelectMSBPerformance (unsigned nBankMSB) +{ + nBankMSB=constrain((int)nBankMSB,0,127); + m_nVoiceBankIDMSBPerformance = nBankMSB; +} + void CMiniDexed::BankSelectLSB (unsigned nBankLSB, unsigned nTG) { nBankLSB=constrain((int)nBankLSB,0,127); @@ -435,6 +476,18 @@ void CMiniDexed::BankSelectLSB (unsigned nBankLSB, unsigned nTG) BankSelect(nBank, nTG); } +void CMiniDexed::BankSelectLSBPerformance (unsigned nBankLSB) +{ + nBankLSB=constrain((int)nBankLSB,0,127); + + unsigned nBank = m_nVoiceBankIDPerformance; + unsigned nBankMSB = m_nVoiceBankIDMSBPerformance; + nBank = (nBankMSB << 7) + nBankLSB; + + // Now should have both MSB and LSB so enable the BankSelect + BankSelectPerformance(nBank); +} + void CMiniDexed::ProgramChange (unsigned nProgram, unsigned nTG) { assert (m_pConfig); @@ -489,10 +542,7 @@ void CMiniDexed::ProgramChangePerformance (unsigned nProgram) if (m_nParameter[ParameterPerformanceSelectChannel] != CMIDIDevice::Disabled) { // Program Change messages change Performances. - unsigned nLastPerformance = m_PerformanceConfig.GetLastPerformance(); - - // GetLastPerformance actually returns 1-indexed, number of performances - if (nProgram < nLastPerformance - 1) + if (m_PerformanceConfig.IsValidPerformance(nProgram)) { SetNewPerformance(nProgram); } @@ -800,6 +850,10 @@ void CMiniDexed::SetParameter (TParameter Parameter, int nValue) // Nothing more to do break; + case ParameterPerformanceBank: + BankSelectPerformance(nValue); + break; + default: assert (0); break; @@ -1181,10 +1235,17 @@ void CMiniDexed::SetPerformanceSelectChannel (unsigned uCh) bool CMiniDexed::SavePerformance (bool bSaveAsDeault) { - m_bSavePerformance = true; - m_bSaveAsDeault=bSaveAsDeault; + if (m_PerformanceConfig.GetInternalFolderOk()) + { + m_bSavePerformance = true; + m_bSaveAsDeault=bSaveAsDeault; - return true; + return true; + } + else + { + return false; + } } bool CMiniDexed::DoSavePerformance (void) @@ -1501,6 +1562,16 @@ unsigned CMiniDexed::GetLastPerformance() return m_PerformanceConfig.GetLastPerformance(); } +unsigned CMiniDexed::GetPerformanceBank() +{ + return m_PerformanceConfig.GetPerformanceBank(); +} + +unsigned CMiniDexed::GetLastPerformanceBank() +{ + return m_PerformanceConfig.GetLastPerformanceBank(); +} + unsigned CMiniDexed::GetActualPerformanceID() { return m_PerformanceConfig.GetActualPerformanceID(); @@ -1511,6 +1582,16 @@ void CMiniDexed::SetActualPerformanceID(unsigned nID) m_PerformanceConfig.SetActualPerformanceID(nID); } +unsigned CMiniDexed::GetActualPerformanceBankID() +{ + return m_PerformanceConfig.GetActualPerformanceBankID(); +} + +void CMiniDexed::SetActualPerformanceBankID(unsigned nBankID) +{ + m_PerformanceConfig.SetActualPerformanceBankID(nBankID); +} + bool CMiniDexed::SetNewPerformance(unsigned nID) { m_bSetNewPerformance = true; @@ -1519,6 +1600,20 @@ bool CMiniDexed::SetNewPerformance(unsigned nID) return true; } +bool CMiniDexed::SetNewPerformanceBank(unsigned nBankID) +{ + m_bSetNewPerformanceBank = true; + m_nSetNewPerformanceBankID = nBankID; + + return true; +} + +void CMiniDexed::SetFirstPerformance(void) +{ + m_bSetFirstPerformance = true; + return; +} + bool CMiniDexed::DoSetNewPerformance (void) { m_bLoadPerformanceBusy = true; @@ -1540,6 +1635,25 @@ bool CMiniDexed::DoSetNewPerformance (void) } } +bool CMiniDexed::DoSetNewPerformanceBank (void) +{ + m_bLoadPerformanceBankBusy = true; + + unsigned nBankID = m_nSetNewPerformanceBankID; + m_PerformanceConfig.SetNewPerformanceBank(nBankID); + + m_bLoadPerformanceBankBusy = false; + return true; +} + +void CMiniDexed::DoSetFirstPerformance(void) +{ + unsigned nID = m_PerformanceConfig.FindFirstPerformance(); + SetNewPerformance(nID); + m_bSetFirstPerformance = false; + return; +} + bool CMiniDexed::SavePerformanceNewFile () { m_bSavePerformanceNewFile = m_PerformanceConfig.GetInternalFolderOk() && m_PerformanceConfig.CheckFreePerformanceSlot(); @@ -1631,6 +1745,16 @@ void CMiniDexed::SetNewPerformanceName(std::string nName) m_PerformanceConfig.SetNewPerformanceName(nName); } +bool CMiniDexed::IsValidPerformance(unsigned nID) +{ + return m_PerformanceConfig.IsValidPerformance(nID); +} + +bool CMiniDexed::IsValidPerformanceBank(unsigned nBankID) +{ + return m_PerformanceConfig.IsValidPerformanceBank(nBankID); +} + void CMiniDexed::SetVoiceName (std::string VoiceName, unsigned nTG) { assert (nTG < CConfig::ToneGenerators); @@ -1642,10 +1766,17 @@ void CMiniDexed::SetVoiceName (std::string VoiceName, unsigned nTG) bool CMiniDexed::DeletePerformance(unsigned nID) { - m_bDeletePerformance = true; - m_nDeletePerformanceID = nID; + if (m_PerformanceConfig.IsValidPerformance(nID) && m_PerformanceConfig.GetInternalFolderOk()) + { + m_bDeletePerformance = true; + m_nDeletePerformanceID = nID; - return true; + return true; + } + else + { + return false; + } } bool CMiniDexed::DoDeletePerformance(void) diff --git a/src/minidexed.h b/src/minidexed.h index 1aa3096..407d5db 100644 --- a/src/minidexed.h +++ b/src/minidexed.h @@ -62,10 +62,14 @@ public: #endif CSysExFileLoader *GetSysExFileLoader (void); + CPerformanceConfig *GetPerformanceConfig (void); void BankSelect (unsigned nBank, unsigned nTG); + void BankSelectPerformance (unsigned nBank); void BankSelectMSB (unsigned nBankMSB, unsigned nTG); + void BankSelectMSBPerformance (unsigned nBankMSB); void BankSelectLSB (unsigned nBankLSB, unsigned nTG); + void BankSelectLSBPerformance (unsigned nBankLSB); void ProgramChange (unsigned nProgram, unsigned nTG); void ProgramChangePerformance (unsigned nProgram); void SetVolume (unsigned nVolume, unsigned nTG); @@ -117,17 +121,27 @@ public: std::string GetPerformanceFileName(unsigned nID); std::string GetPerformanceName(unsigned nID); unsigned GetLastPerformance(); + unsigned GetPerformanceBank(); + unsigned GetLastPerformanceBank(); unsigned GetActualPerformanceID(); void SetActualPerformanceID(unsigned nID); + unsigned GetActualPerformanceBankID(); + void SetActualPerformanceBankID(unsigned nBankID); bool SetNewPerformance(unsigned nID); + bool SetNewPerformanceBank(unsigned nBankID); + void SetFirstPerformance(void); + void DoSetFirstPerformance(void); bool SavePerformanceNewFile (); bool DoSavePerformanceNewFile (void); bool DoSetNewPerformance (void); + bool DoSetNewPerformanceBank (void); bool GetPerformanceSelectToLoad(void); bool SavePerformance (bool bSaveAsDeault); unsigned GetPerformanceSelectChannel (void); void SetPerformanceSelectChannel (unsigned uCh); + bool IsValidPerformance(unsigned nID); + bool IsValidPerformanceBank(unsigned nBankID); // Must match the order in CUIMenu::TParameter enum TParameter @@ -141,6 +155,7 @@ public: ParameterReverbDiffusion, ParameterReverbLevel, ParameterPerformanceSelectChannel, + ParameterPerformanceBank, ParameterUnknown }; @@ -238,6 +253,8 @@ private: unsigned m_nVoiceBankID[CConfig::ToneGenerators]; unsigned m_nVoiceBankIDMSB[CConfig::ToneGenerators]; + unsigned m_nVoiceBankIDPerformance; + unsigned m_nVoiceBankIDMSBPerformance; unsigned m_nProgram[CConfig::ToneGenerators]; unsigned m_nVolume[CConfig::ToneGenerators]; unsigned m_nPan[CConfig::ToneGenerators]; @@ -305,9 +322,13 @@ private: bool m_bSavePerformanceNewFile; bool m_bSetNewPerformance; unsigned m_nSetNewPerformanceID; + bool m_bSetNewPerformanceBank; + unsigned m_nSetNewPerformanceBankID; + bool m_bSetFirstPerformance; bool m_bDeletePerformance; unsigned m_nDeletePerformanceID; bool m_bLoadPerformanceBusy; + bool m_bLoadPerformanceBankBusy; bool m_bSaveAsDeault; }; diff --git a/src/performanceconfig.cpp b/src/performanceconfig.cpp index 853ccf5..8cd6275 100644 --- a/src/performanceconfig.cpp +++ b/src/performanceconfig.cpp @@ -28,8 +28,15 @@ LOGMODULE ("Performance"); +//#define VERBOSE_DEBUG + +#define PERFORMANCE_DIR "performance" +#define DEFAULT_PERFORMANCE_FILENAME "performance.ini" +#define DEFAULT_PERFORMANCE_NAME "Default" +#define DEFAULT_PERFORMANCE_BANK_NAME "Default" + CPerformanceConfig::CPerformanceConfig (FATFS *pFileSystem) -: m_Properties ("performance.ini", pFileSystem) +: m_Properties (DEFAULT_PERFORMANCE_FILENAME, pFileSystem) { m_pFileSystem = pFileSystem; } @@ -38,6 +45,47 @@ CPerformanceConfig::~CPerformanceConfig (void) { } +bool CPerformanceConfig::Init (void) +{ + // Check intermal performance directory exists + DIR Directory; + FRESULT Result; + //Check if internal "performance" directory exists + Result = f_opendir (&Directory, "SD:/" PERFORMANCE_DIR); + if (Result == FR_OK) + { + m_bPerformanceDirectoryExists=true; + Result = f_closedir (&Directory); + } + else + { + m_bPerformanceDirectoryExists = false; + } + + // List banks if present + ListPerformanceBanks(); + +#ifdef VERBOSE_DEBUG +#warning "PerformanceConfig in verbose debug printing mode" + LOGNOTE("Testing loading of banks"); + for (unsigned i=0; i= NUM_PERFORMANCES) { + if (!m_bPerformanceDirectoryExists) + { + // Nothing can be done if there is no performance directory + LOGNOTE("Performance directory does not exist"); + return false; + } + if (m_nLastPerformance >= NUM_PERFORMANCES) { // No space left for new performances LOGWARN ("No space left for new performance"); return false; } + + // Note: New performances are created at the end of the currently selected bank. + // Sould we default to a new bank just for user-performances? + // + // There is the possibility of MIDI changing the Bank Number and the user + // not spotting the bank has changed... + // + // Another option would be to find empty slots in the current bank + // rather than always add at the end. + // + // Sorting this out is left for a future update. std::string sPerformanceName = NewPerformanceName; NewPerformanceName=""; - nActualPerformance=nLastPerformance; + unsigned nNewPerformance = m_nLastPerformance + 1; std::string nFileName; std::string nPath; std::string nIndex = "000000"; - nIndex += std::to_string(++nLastFileIndex); + nIndex += std::to_string(nNewPerformance+1); // Index on disk = index in memory+1 nIndex = nIndex.substr(nIndex.length()-6,6); @@ -793,10 +930,11 @@ bool CPerformanceConfig::CreateNewPerformanceFile(void) nFileName +=sPerformanceName.substr(0,14); } nFileName += ".ini"; - m_nPerformanceFileName[nLastPerformance]= nFileName; + m_PerformanceFileName[nNewPerformance]= sPerformanceName; nPath = "SD:/" ; nPath += PERFORMANCE_DIR; + nPath += AddPerformanceBankDirName(m_nPerformanceBank); nPath += "/"; nFileName = nPath + nFileName; @@ -804,17 +942,18 @@ bool CPerformanceConfig::CreateNewPerformanceFile(void) FRESULT Result = f_open (&File, nFileName.c_str(), FA_WRITE | FA_CREATE_ALWAYS); if (Result != FR_OK) { - m_nPerformanceFileName[nLastPerformance]=nullptr; + m_PerformanceFileName[nNewPerformance]=nullptr; return false; } if (f_close (&File) != FR_OK) { - m_nPerformanceFileName[nLastPerformance]=nullptr; + m_PerformanceFileName[nNewPerformance]=nullptr; return false; } - nLastPerformance++; + m_nLastPerformance = nNewPerformance; + m_nActualPerformance = nNewPerformance; new (&m_Properties) CPropertiesFatFsFile(nFileName.c_str(), m_pFileSystem); return true; @@ -822,88 +961,124 @@ bool CPerformanceConfig::CreateNewPerformanceFile(void) bool CPerformanceConfig::ListPerformances() { - nInternalFolderOk=false; - nExternalFolderOk=false; // for future USB implementation - nLastPerformance=0; - nLastFileIndex=0; - m_nPerformanceFileName[nLastPerformance++]="performance.ini"; // in order to assure retrocompatibility - - unsigned nPIndex; - DIR Directory; - FILINFO FileInfo; - FRESULT Result; - //Check if internal "performance" directory exists - Result = f_opendir (&Directory, "SD:/" PERFORMANCE_DIR); - if (Result == FR_OK) + // Clear any existing lists of performances + for (unsigned i=0; i= NUM_PERFORMANCES) { - LOGNOTE ("Skipping performance %s", FileInfo.fname); - } else { - if (!(FileInfo.fattrib & (AM_HID | AM_SYS))) + if (!(FileInfo.fattrib & (AM_HID | AM_SYS))) + { + std::string OriFileName = FileInfo.fname; + size_t nLen = OriFileName.length(); + if ( nLen > 8 && nLen <26 && strcmp(OriFileName.substr(6,1).c_str(), "_")==0) { - std::string FileName = FileInfo.fname; - size_t nLen = FileName.length(); - if ( nLen > 8 && nLen <26 && strcmp(FileName.substr(6,1).c_str(), "_")==0) - { - nPIndex=stoi(FileName.substr(0,6)); - if(nPIndex > nLastFileIndex) + // Note: m_nLastPerformance - refers to the number (index) of the last performance in memory, + // which includes a default performance. + // + // Filenames on the disk start from 1 to match what the user might see in MIDI. + // So this means that actually file 000001_ will correspond to index position [0]. + // For the default bank though, ID 1 is the default performance, so will already exist. + // m_PerformanceFileName[0] = default performance (file 000001) + // m_PerformanceFileName[1] = first available on-disk performance (file 000002) + // + // Note2: filenames assume 6 digits, underscore, name, finally ".ini" + // i.e. 123456_Performance Name.ini + // + nPIndex=stoi(OriFileName.substr(0,6)); + if ((nPIndex < 1) || (nPIndex >= (NUM_PERFORMANCES+1))) + { + // Index is out of range - skip to next file + LOGNOTE ("Performance number out of range: %s (%d to %d)", FileInfo.fname, 1, NUM_PERFORMANCES); + } + else + { + // Convert from "user facing" 1..indexed number to internal 0..indexed + nPIndex = nPIndex-1; + if (m_PerformanceFileName[nPIndex].empty()) { - nLastFileIndex=nPIndex; - } + if(nPIndex > m_nLastPerformance) + { + m_nLastPerformance=nPIndex; + } - m_nPerformanceFileName[nLastPerformance++]= FileName; - } + std::string FileName = OriFileName.substr(0,OriFileName.length()-4).substr(7,14); + + m_PerformanceFileName[nPIndex] = FileName; +#ifdef VERBOSE_DEBUG + LOGNOTE ("Loading performance %s (%d, %s)", OriFileName.c_str(), nPIndex, FileName.c_str()); +#endif + } + else + { + LOGNOTE ("Duplicate performance %s", OriFileName.c_str()); + } + } } } Result = f_findnext (&Directory, &FileInfo); } - // sort by performance number-name - if (nLastPerformance > 2) - { - sort (m_nPerformanceFileName+1, m_nPerformanceFileName + nLastPerformance); // default is always on first place. %%%%%%%%%%%%%%%% - } + f_closedir (&Directory); } - LOGNOTE ("Number of Performances: %d", nLastPerformance); - - return nInternalFolderOk; -} - + return true; +} void CPerformanceConfig::SetNewPerformance (unsigned nID) { - nActualPerformance=nID; - std::string FileN = ""; - if (nID != 0) // in order to assure retrocompatibility + assert (nID < NUM_PERFORMANCES); + m_nActualPerformance=nID; + std::string FileN = GetPerformanceFullFilePath(nID); + + new (&m_Properties) CPropertiesFatFsFile(FileN.c_str(), m_pFileSystem); +#ifdef VERBOSE_DEBUG + LOGNOTE("Selecting Performance: %d (%s)", nID+1, FileN.c_str()); +#endif +} + +unsigned CPerformanceConfig::FindFirstPerformance (void) +{ + for (int nID=0; nID < NUM_PERFORMANCES; nID++) + { + if (IsValidPerformance(nID)) { - FileN += PERFORMANCE_DIR; - FileN += "/"; + return nID; } - FileN += m_nPerformanceFileName[nID]; - new (&m_Properties) CPropertiesFatFsFile(FileN.c_str(), m_pFileSystem); - + } + + return 0; // Even though 0 is a valid performance, not much else to do } std::string CPerformanceConfig::GetNewPerformanceDefaultName(void) { std::string nIndex = "000000"; - nIndex += std::to_string(nLastFileIndex+1); + nIndex += std::to_string(m_nLastPerformance+1+1); // Convert from internal 0.. index to a file-based 1.. index to show the user nIndex = nIndex.substr(nIndex.length()-6,6); return "Perf" + nIndex; } @@ -923,31 +1098,229 @@ void CPerformanceConfig::SetNewPerformanceName(std::string nName) bool CPerformanceConfig::DeletePerformance(unsigned nID) { + if (!m_bPerformanceDirectoryExists) + { + // Nothing can be done if there is no performance directory + LOGNOTE("Performance directory does not exist"); + return false; + } bool bOK = false; - if(nID == 0){return bOK;} // default (performance.ini at root directory) can't be deleted + if((m_nPerformanceBank == 0) && (nID == 0)){return bOK;} // default (performance.ini at root directory) can't be deleted DIR Directory; FILINFO FileInfo; std::string FileN = "SD:/"; FileN += PERFORMANCE_DIR; - + FileN += AddPerformanceBankDirName(m_nPerformanceBank); - FRESULT Result = f_findfirst (&Directory, &FileInfo, FileN.c_str(), m_nPerformanceFileName[nID].c_str()); + FRESULT Result = f_findfirst (&Directory, &FileInfo, FileN.c_str(), GetPerformanceFileName(nID).c_str()); if (Result == FR_OK && FileInfo.fname[0]) { FileN += "/"; - FileN += m_nPerformanceFileName[nID]; + FileN += GetPerformanceFileName(nID); Result=f_unlink (FileN.c_str()); if (Result == FR_OK) { SetNewPerformance(0); - nActualPerformance =0; + m_nActualPerformance =0; //nMenuSelectedPerformance=0; - m_nPerformanceFileName[nID]="ZZZZZZ"; - sort (m_nPerformanceFileName+1, m_nPerformanceFileName + nLastPerformance); // test si va con -1 o no - --nLastPerformance; - m_nPerformanceFileName[nLastPerformance]=nullptr; + m_PerformanceFileName[nID].clear(); + // If this was the last performance in the bank... + if (nID == m_nLastPerformance) + { + do + { + // Find the new last performance + m_nLastPerformance--; + } while (!IsValidPerformance(m_nLastPerformance) && (m_nLastPerformance > 0)); + } bOK=true; } + else + { + LOGNOTE ("Failed to delete %s", FileN.c_str()); + } } return bOK; } + +bool CPerformanceConfig::ListPerformanceBanks() +{ + m_nPerformanceBank = 0; + m_nLastPerformance = 0; + m_nLastPerformanceBank = 0; + + // Open performance directory + DIR Directory; + FILINFO FileInfo; + FRESULT Result; + Result = f_opendir (&Directory, "SD:/" PERFORMANCE_DIR); + if (Result != FR_OK) + { + // No performance directory, so no performance banks. + // So nothing else to do here + LOGNOTE ("No performance banks detected"); + m_bPerformanceDirectoryExists = false; + return false; + } + + unsigned nNumBanks = 0; + + // Add in the default performance directory as the first bank + m_PerformanceBankName[0] = DEFAULT_PERFORMANCE_BANK_NAME; + nNumBanks = 1; + m_nLastPerformanceBank = 0; + + // List directories with names in format 01_Perf Bank Name + Result = f_findfirst (&Directory, &FileInfo, "SD:/" PERFORMANCE_DIR, "*"); + for (unsigned i = 0; Result == FR_OK && FileInfo.fname[0]; i++) + { + // Check to see if it is a directory + if ((FileInfo.fattrib & AM_DIR) != 0) + { + // Looking for Performance banks of the form: 01_Perf Bank Name + // So positions 0,1,2 = decimal digit + // position 3 = "_" + // positions 4.. = actual name + // + std::string OriFileName = FileInfo.fname; + size_t nLen = OriFileName.length(); + if ( nLen > 4 && nLen <26 && strcmp(OriFileName.substr(3,1).c_str(), "_")==0) + { + unsigned nBankIndex = stoi(OriFileName.substr(0,3)); + // Recall user index numbered 002..NUM_PERFORMANCE_BANKS + // NB: Bank 001 is reserved for the default performance directory + if ((nBankIndex > 0) && (nBankIndex <= NUM_PERFORMANCE_BANKS)) + { + // Convert from "user facing" 1..indexed number to internal 0..indexed + nBankIndex = nBankIndex-1; + if (m_PerformanceBankName[nBankIndex].empty()) + { + std::string BankName = OriFileName.substr(4,nLen); + + m_PerformanceBankName[nBankIndex] = BankName; +#ifdef VERBOSE_DEBUG + LOGNOTE ("Found performance bank %s (%d, %s)", OriFileName.c_str(), nBankIndex, BankName.c_str()); +#endif + nNumBanks++; + if (nBankIndex > m_nLastPerformanceBank) + { + m_nLastPerformanceBank = nBankIndex; + } + } + else + { + LOGNOTE ("Duplicate Performance Bank: %s", FileInfo.fname); + if (nBankIndex==0) + { + LOGNOTE ("(Bank 001 is the default performance directory)"); + } + } + } + else + { + LOGNOTE ("Performance Bank number out of range: %s (%d to %d)", FileInfo.fname, 1, NUM_PERFORMANCE_BANKS); + } + } + else + { +#ifdef VERBOSE_DEBUG + LOGNOTE ("Skipping: %s", FileInfo.fname); +#endif + } + } + + Result = f_findnext (&Directory, &FileInfo); + } + + if (nNumBanks > 0) + { + LOGNOTE ("Number of Performance Banks: %d (last = %d)", nNumBanks, m_nLastPerformanceBank+1); + } + + f_closedir (&Directory); + return true; +} + +void CPerformanceConfig::SetNewPerformanceBank(unsigned nBankID) +{ + assert (nBankID < NUM_PERFORMANCE_BANKS); + if (IsValidPerformanceBank(nBankID)) + { +#ifdef VERBOSE_DEBUG + LOGNOTE("Selecting Performance Bank: %d", nBankID+1); +#endif + m_nPerformanceBank = nBankID; + m_nActualPerformanceBank = nBankID; + ListPerformances(); + } + else + { +#ifdef VERBOSE_DEBUG + LOGNOTE("Not selecting invalid Performance Bank: %d", nBankID+1); +#endif + } +} + +unsigned CPerformanceConfig::GetPerformanceBank(void) +{ + return m_nPerformanceBank; +} + +std::string CPerformanceConfig::GetPerformanceBankName(unsigned nBankID) +{ + assert (nBankID < NUM_PERFORMANCE_BANKS); + if (IsValidPerformanceBank(nBankID)) + { + return m_PerformanceBankName[nBankID]; + } + else + { + return DEFAULT_PERFORMANCE_BANK_NAME; + } +} + +std::string CPerformanceConfig::AddPerformanceBankDirName(unsigned nBankID) +{ + assert (nBankID < NUM_PERFORMANCE_BANKS); + if (IsValidPerformanceBank(nBankID)) + { + // Performance Banks directories in format "001_Bank Name" + std::string Index; + if (nBankID == 0) + { + // Legacy: Bank 1 is the default performance directory + return ""; + } + + if (nBankID < 9) + { + Index = "00" + std::to_string(nBankID+1); + } + else if (nBankID < 99) + { + Index = "0" + std::to_string(nBankID+1); + } + else + { + Index = std::to_string(nBankID+1); + } + + return "/" + Index + "_" + m_PerformanceBankName[nBankID]; + } + else + { + return ""; + } +} + +bool CPerformanceConfig::IsValidPerformanceBank(unsigned nBankID) +{ + if (nBankID >= NUM_PERFORMANCE_BANKS) { + return false; + } + if (m_PerformanceBankName[nBankID].empty()) + { + return false; + } + return true; +} diff --git a/src/performanceconfig.h b/src/performanceconfig.h index c151c7a..0d50daa 100644 --- a/src/performanceconfig.h +++ b/src/performanceconfig.h @@ -27,14 +27,16 @@ #include #include #define NUM_VOICE_PARAM 156 -#define PERFORMANCE_DIR "performance" -#define NUM_PERFORMANCES 256 +#define NUM_PERFORMANCES 128 +#define NUM_PERFORMANCE_BANKS 128 class CPerformanceConfig // Performance configuration { public: CPerformanceConfig (FATFS *pFileSystem); ~CPerformanceConfig (void); + + bool Init (void); bool Load (void); @@ -122,17 +124,30 @@ public: bool ListPerformances(); //std::string m_DirName; void SetNewPerformance (unsigned nID); + unsigned FindFirstPerformance (void); std::string GetPerformanceFileName(unsigned nID); + std::string GetPerformanceFullFilePath(unsigned nID); std::string GetPerformanceName(unsigned nID); unsigned GetLastPerformance(); + unsigned GetLastPerformanceBank(); void SetActualPerformanceID(unsigned nID); unsigned GetActualPerformanceID(); + void SetActualPerformanceBankID(unsigned nBankID); + unsigned GetActualPerformanceBankID(); bool CreateNewPerformanceFile(void); bool GetInternalFolderOk(); std::string GetNewPerformanceDefaultName(void); void SetNewPerformanceName(std::string nName); bool DeletePerformance(unsigned nID); bool CheckFreePerformanceSlot(void); + std::string AddPerformanceBankDirName(unsigned nBankID); + bool IsValidPerformance(unsigned nID); + + bool ListPerformanceBanks(void); + void SetNewPerformanceBank(unsigned nBankID); + unsigned GetPerformanceBank(void); + std::string GetPerformanceBankName(unsigned nBankID); + bool IsValidPerformanceBank(unsigned nBankID); private: CPropertiesFatFsFile m_Properties; @@ -166,15 +181,17 @@ private: unsigned m_nAftertouchRange[CConfig::ToneGenerators]; unsigned m_nAftertouchTarget[CConfig::ToneGenerators]; - unsigned nLastPerformance; - unsigned nLastFileIndex; - unsigned nActualPerformance = 0; + unsigned m_nLastPerformance; + unsigned m_nActualPerformance = 0; + unsigned m_nActualPerformanceBank = 0; + unsigned m_nPerformanceBank; + unsigned m_nLastPerformanceBank; + bool m_bPerformanceDirectoryExists; //unsigned nMenuSelectedPerformance = 0; - std::string m_nPerformanceFileName[NUM_PERFORMANCES]; + std::string m_PerformanceFileName[NUM_PERFORMANCES]; + std::string m_PerformanceBankName[NUM_PERFORMANCE_BANKS]; FATFS *m_pFileSystem; - bool nInternalFolderOk=false; - bool nExternalFolderOk=false; // for future USB implementation std::string NewPerformanceName=""; bool m_bCompressorEnable; diff --git a/src/uimenu.cpp b/src/uimenu.cpp index 93e6b0e..82a426a 100644 --- a/src/uimenu.cpp +++ b/src/uimenu.cpp @@ -214,7 +214,8 @@ const CUIMenu::TParameter CUIMenu::s_GlobalParameter[CMiniDexed::ParameterUnknow {0, 99, 1}, // ParameterReverbLowPass {0, 99, 1}, // ParameterReverbDiffusion {0, 99, 1}, // ParameterReverbLevel - {0, CMIDIDevice::ChannelUnknown-1, 1, ToMIDIChannel} // ParameterPerformanceSelectChannel + {0, CMIDIDevice::ChannelUnknown-1, 1, ToMIDIChannel}, // ParameterPerformanceSelectChannel + {0, NUM_PERFORMANCE_BANKS, 1} // ParameterPerformanceBank }; // must match CMiniDexed::TTGParameter @@ -327,6 +328,7 @@ const CUIMenu::TMenuItem CUIMenu::s_PerformanceMenu[] = {"Load", PerformanceMenu, 0, 0}, {"Save", MenuHandler, s_SaveMenu}, {"Delete", PerformanceMenu, 0, 1}, + {"Bank", EditPerformanceBankNumber, 0, 0}, {"PCCH", EditGlobalParameter, 0, CMiniDexed::ParameterPerformanceSelectChannel}, {0} }; @@ -1211,24 +1213,43 @@ void CUIMenu::PgmUpDownHandler (TMenuEvent Event) // Program Up/Down acts on performances unsigned nLastPerformance = m_pMiniDexed->GetLastPerformance(); unsigned nPerformance = m_pMiniDexed->GetActualPerformanceID(); + unsigned nStart = nPerformance; //LOGNOTE("Performance actual=%d, last=%d", nPerformance, nLastPerformance); if (Event == MenuEventPgmDown) { - if (nPerformance > 0) + do { - m_nSelectedPerformanceID = nPerformance-1; - m_pMiniDexed->SetNewPerformance(m_nSelectedPerformanceID); - //LOGNOTE("Performance new=%d, last=%d", m_nSelectedPerformanceID, nLastPerformance); - } + if (nPerformance == 0) + { + // Wrap around + nPerformance = nLastPerformance; + } + else if (nPerformance > 0) + { + --nPerformance; + } + } while ((m_pMiniDexed->IsValidPerformance(nPerformance) != true) && (nPerformance != nStart)); + m_nSelectedPerformanceID = nPerformance; + m_pMiniDexed->SetNewPerformance(m_nSelectedPerformanceID); + //LOGNOTE("Performance new=%d, last=%d", m_nSelectedPerformanceID, nLastPerformance); } - else + else // MenuEventPgmUp { - if (nPerformance < nLastPerformance-1) + do { - m_nSelectedPerformanceID = nPerformance+1; - m_pMiniDexed->SetNewPerformance(m_nSelectedPerformanceID); - //LOGNOTE("Performance new=%d, last=%d", m_nSelectedPerformanceID, nLastPerformance); - } + if (nPerformance == nLastPerformance) + { + // Wrap around + nPerformance = 0; + } + else if (nPerformance < nLastPerformance) + { + ++nPerformance; + } + } while ((m_pMiniDexed->IsValidPerformance(nPerformance) != true) && (nPerformance != nStart)); + m_nSelectedPerformanceID = nPerformance; + m_pMiniDexed->SetNewPerformance(m_nSelectedPerformanceID); + //LOGNOTE("Performance new=%d, last=%d", m_nSelectedPerformanceID, nLastPerformance); } } else @@ -1366,7 +1387,15 @@ void CUIMenu::TimerHandlerNoBack (TKernelTimerHandle hTimer, void *pParam, void void CUIMenu::PerformanceMenu (CUIMenu *pUIMenu, TMenuEvent Event) { bool bPerformanceSelectToLoad = pUIMenu->m_pMiniDexed->GetPerformanceSelectToLoad(); + unsigned nLastPerformance = pUIMenu->m_pMiniDexed->GetLastPerformance(); unsigned nValue = pUIMenu->m_nSelectedPerformanceID; + unsigned nStart = nValue; + if (pUIMenu->m_pMiniDexed->IsValidPerformance(nValue) != true) + { + // A bank change has left the selected performance out of sync + nValue = pUIMenu->m_pMiniDexed->GetActualPerformanceID(); + pUIMenu->m_nSelectedPerformanceID = nValue; + } std::string Value; if (Event == MenuEventUpdate) @@ -1380,17 +1409,25 @@ void CUIMenu::PerformanceMenu (CUIMenu *pUIMenu, TMenuEvent Event) } if(!pUIMenu->m_bPerformanceDeleteMode) - { + { switch (Event) { case MenuEventUpdate: break; case MenuEventStepDown: - if (nValue > 0) + do { - --nValue; - } + if (nValue == 0) + { + // Wrap around + nValue = nLastPerformance; + } + else if (nValue > 0) + { + --nValue; + } + } while ((pUIMenu->m_pMiniDexed->IsValidPerformance(nValue) != true) && (nValue != nStart)); pUIMenu->m_nSelectedPerformanceID = nValue; if (!bPerformanceSelectToLoad && pUIMenu->m_nCurrentParameter==0) { @@ -1399,10 +1436,18 @@ void CUIMenu::PerformanceMenu (CUIMenu *pUIMenu, TMenuEvent Event) break; case MenuEventStepUp: - if (++nValue > (unsigned) pUIMenu->m_pMiniDexed->GetLastPerformance()-1) + do { - nValue = pUIMenu->m_pMiniDexed->GetLastPerformance()-1; - } + if (nValue == nLastPerformance) + { + // Wrap around + nValue = 0; + } + else if (nValue < nLastPerformance) + { + ++nValue; + } + } while ((pUIMenu->m_pMiniDexed->IsValidPerformance(nValue) != true) && (nValue != nStart)); pUIMenu->m_nSelectedPerformanceID = nValue; if (!bPerformanceSelectToLoad && pUIMenu->m_nCurrentParameter==0) { @@ -1421,7 +1466,7 @@ void CUIMenu::PerformanceMenu (CUIMenu *pUIMenu, TMenuEvent Event) break; case 1: - if (pUIMenu->m_nSelectedPerformanceID != 0) + if (pUIMenu->m_pMiniDexed->IsValidPerformance(pUIMenu->m_nSelectedPerformanceID)) { pUIMenu->m_bPerformanceDeleteMode=true; pUIMenu->m_bConfirmDeletePerformance=false; @@ -1474,17 +1519,24 @@ void CUIMenu::PerformanceMenu (CUIMenu *pUIMenu, TMenuEvent Event) if(!pUIMenu->m_bPerformanceDeleteMode) { Value = pUIMenu->m_pMiniDexed->GetPerformanceName(nValue); + unsigned nBankNum = pUIMenu->m_pMiniDexed->GetPerformanceBank(); - - std::string nPSelected = ""; + std::string nPSelected = "000"; + nPSelected += std::to_string(nBankNum+1); // Convert to user-facing bank number rather than index + nPSelected = nPSelected.substr(nPSelected.length()-3,3); + std::string nPPerf = "000"; + nPPerf += std::to_string(nValue+1); // Convert to user-facing performance number rather than index + nPPerf = nPPerf.substr(nPPerf.length()-3,3); + + nPSelected += ":"+nPPerf; if(nValue == pUIMenu->m_pMiniDexed->GetActualPerformanceID()) { - nPSelected= "[L]"; + nPSelected += " [L]"; } pUIMenu->m_pUI->DisplayWrite (pUIMenu->m_pParentMenu[pUIMenu->m_nCurrentMenuItem].Name, nPSelected.c_str(), - Value.c_str (), - (int) nValue > 0, (int) nValue < (int) pUIMenu->m_pMiniDexed->GetLastPerformance()-1); + Value.c_str (), true, true); +// (int) nValue > 0, (int) nValue < (int) pUIMenu->m_pMiniDexed->GetLastPerformance()); } else { @@ -1492,6 +1544,90 @@ void CUIMenu::PerformanceMenu (CUIMenu *pUIMenu, TMenuEvent Event) } } +void CUIMenu::EditPerformanceBankNumber (CUIMenu *pUIMenu, TMenuEvent Event) +{ + bool bPerformanceSelectToLoad = pUIMenu->m_pMiniDexed->GetPerformanceSelectToLoad(); + unsigned nLastPerformanceBank = pUIMenu->m_pMiniDexed->GetLastPerformanceBank(); + unsigned nValue = pUIMenu->m_nSelectedPerformanceBankID; + unsigned nStart = nValue; + std::string Value; + + switch (Event) + { + case MenuEventUpdate: + break; + + case MenuEventStepDown: + do + { + if (nValue == 0) + { + // Wrap around + nValue = nLastPerformanceBank; + } + else if (nValue > 0) + { + --nValue; + } + } while ((pUIMenu->m_pMiniDexed->IsValidPerformanceBank(nValue) != true) && (nValue != nStart)); + pUIMenu->m_nSelectedPerformanceBankID = nValue; + if (!bPerformanceSelectToLoad) + { + // Switch to the new bank and select the first performance voice + pUIMenu->m_pMiniDexed->SetParameter (CMiniDexed::ParameterPerformanceBank, nValue); + pUIMenu->m_pMiniDexed->SetFirstPerformance(); + } + break; + + case MenuEventStepUp: + do + { + if (nValue == nLastPerformanceBank) + { + // Wrap around + nValue = 0; + } + else if (nValue < nLastPerformanceBank) + { + ++nValue; + } + } while ((pUIMenu->m_pMiniDexed->IsValidPerformanceBank(nValue) != true) && (nValue != nStart)); + pUIMenu->m_nSelectedPerformanceBankID = nValue; + if (!bPerformanceSelectToLoad) + { + pUIMenu->m_pMiniDexed->SetParameter (CMiniDexed::ParameterPerformanceBank, nValue); + pUIMenu->m_pMiniDexed->SetFirstPerformance(); + } + break; + + case MenuEventSelect: + if (bPerformanceSelectToLoad) + { + pUIMenu->m_pMiniDexed->SetParameter (CMiniDexed::ParameterPerformanceBank, nValue); + pUIMenu->m_pMiniDexed->SetFirstPerformance(); + } + break; + + default: + return; + } + + Value = pUIMenu->m_pMiniDexed->GetPerformanceConfig ()->GetPerformanceBankName(nValue); + std::string nPSelected = "000"; + nPSelected += std::to_string(nValue+1); // Convert to user-facing number rather than index + nPSelected = nPSelected.substr(nPSelected.length()-3,3); + + if(nValue == (unsigned)pUIMenu->m_pMiniDexed->GetParameter (CMiniDexed::ParameterPerformanceBank)) + { + nPSelected += " [L]"; + } + + pUIMenu->m_pUI->DisplayWrite (pUIMenu->m_pParentMenu[pUIMenu->m_nCurrentMenuItem].Name, nPSelected.c_str(), + Value.c_str (), + nValue > 0, + nValue < pUIMenu->m_pMiniDexed->GetLastPerformanceBank()-1); +} + void CUIMenu::InputTxt (CUIMenu *pUIMenu, TMenuEvent Event) { unsigned nTG=0; diff --git a/src/uimenu.h b/src/uimenu.h index b66d65c..d5b48dc 100644 --- a/src/uimenu.h +++ b/src/uimenu.h @@ -91,6 +91,7 @@ private: static void EditTGParameterModulation (CUIMenu *pUIMenu, TMenuEvent Event); static void PerformanceMenu (CUIMenu *pUIMenu, TMenuEvent Event); static void SavePerformanceNewFile (CUIMenu *pUIMenu, TMenuEvent Event); + static void EditPerformanceBankNumber (CUIMenu *pUIMenu, TMenuEvent Event); static std::string GetGlobalValueString (unsigned nParameter, int nValue); static std::string GetTGValueString (unsigned nTGParameter, int nValue); @@ -169,6 +170,7 @@ private: bool m_bPerformanceDeleteMode=false; bool m_bConfirmDeletePerformance=false; unsigned m_nSelectedPerformanceID =0; + unsigned m_nSelectedPerformanceBankID =0; bool m_bSplashShow=false; }; diff --git a/src/userinterface.cpp b/src/userinterface.cpp index 1c0716e..a2d78f1 100644 --- a/src/userinterface.cpp +++ b/src/userinterface.cpp @@ -388,13 +388,16 @@ void CUserInterface::UISetMIDIButtonChannel (unsigned uCh) if (uCh == 0) { m_nMIDIButtonCh = CMIDIDevice::Disabled; + LOGNOTE("MIDI Button channel not set"); } else if (uCh < CMIDIDevice::Channels) { m_nMIDIButtonCh = uCh - 1; + LOGNOTE("MIDI Button channel set to: %d", m_nMIDIButtonCh); } else { m_nMIDIButtonCh = CMIDIDevice::OmniMode; + LOGNOTE("MIDI Button channel set to: OMNI"); } } \ No newline at end of file From 891a57809c098d77374558baa49141144771eb14 Mon Sep 17 00:00:00 2001 From: Geo Maciolek Date: Mon, 29 Jan 2024 16:21:39 -0500 Subject: [PATCH 02/38] Wiki (anchor) links fixed, markdown linting (#607) "*" lists changed to "-" lists, bare URLs changd to use <...>, alt text added, etc. --- README.md | 76 +++++++++++++++++++++++++++---------------------------- 1 file changed, 38 insertions(+), 38 deletions(-) diff --git a/README.md b/README.md index 8623d9b..a0e6ae5 100644 --- a/README.md +++ b/README.md @@ -1,15 +1,15 @@ -# MiniDexed ![](https://github.com/probonopd/MiniDexed/actions/workflows/build.yml/badge.svg) +# MiniDexed ![Github Build Status](https://github.com/probonopd/MiniDexed/actions/workflows/build.yml/badge.svg) ![minidexed](https://user-images.githubusercontent.com/2480569/161813414-bb156a1c-efec-44c0-802a-8926412a08e0.jpg) -MiniDexed is a FM synthesizer closely modeled on the famous DX7 by a well-known Japanese manufacturer running on a bare metal Raspberry Pi (without a Linux kernel or operating system). On Raspberry Pi 2 and larger, it can run 8 tone generators, not unlike the TX816/TX802 (8 DX7 instances without the keyboard in one box). [Featured by HACKADAY](https://hackaday.com/2022/04/19/bare-metal-gives-this-pi-some-classic-synths/), [adafruit](https://blog.adafruit.com/2022/04/25/free-yamaha-dx7-synth-emulator-on-a-raspberry-pi/), and [Synth Geekery](https://www.youtube.com/watch?v=TDSy5nnm0jA). +MiniDexed is a FM synthesizer closely modeled on the famous DX7 by a well-known Japanese manufacturer running on a bare metal Raspberry Pi (without a Linux kernel or operating system). On Raspberry Pi 2 and larger, it can run 8 tone generators, not unlike the TX816/TX802 (8 DX7 instances without the keyboard in one box). [Featured by HACKADAY](https://hackaday.com/2022/04/19/bare-metal-gives-this-pi-some-classic-synths/), [Adafruit](https://blog.adafruit.com/2022/04/25/free-yamaha-dx7-synth-emulator-on-a-raspberry-pi/), and [Synth Geekery](https://www.youtube.com/watch?v=TDSy5nnm0jA). ## Features - [x] Uses [Synth_Dexed](https://codeberg.org/dcoredump/Synth_Dexed) with [circle-stdlib](https://github.com/smuehlst/circle-stdlib) - [x] SD card contents can be downloaded from [GitHub Releases](../../releases) - [x] Runs on all Raspberry Pi models (except Pico); see below for details -- [x] Produces sound on the headphone jack, HDMI display or [audio extractor](https://github.com/probonopd/MiniDexed/wiki/Hardware#hdmi-to-audio) (better), or a [dedicated DAC](https://github.com/probonopd/MiniDexed/wiki/Hardware#i2c-dac) (best) +- [x] Produces sound on the headphone jack, HDMI display or [audio extractor](https://github.com/probonopd/MiniDexed/wiki/Hardware#hdmi-to-audio) (better), or a [dedicated DAC](https://github.com/probonopd/MiniDexed/wiki/Hardware#i2s-dac) (best) - [x] Supports multiple voices through Program Change and Bank Change LSB/MSB MIDI messages - [x] Loads voices from `.syx` files from SD card (e.g., using `getsysex.sh` or from [Dexed_cart_1.0.zip](http://hsjp.eu/downloads/Dexed/Dexed_cart_1.0.zip)) - [x] Menu structure on optional [HD44780 display](https://www.berrybase.de/sensoren-module/displays/alphanumerische-displays/alphanumerisches-lcd-16x2-gr-252-n/gelb) and rotary encoder @@ -18,48 +18,48 @@ MiniDexed is a FM synthesizer closely modeled on the famous DX7 by a well-known - [x] Allows to configure multiple Dexed instances through `performance.ini` files (e.g., [converted](https://github.com/BobanSpasic/MDX_Vault) from DX1, DX5, TX816, DX7II, TX802) - [x] Compressor effect - [x] Reverb effect -- [x] Voices can be edited over MIDI, e.g., using the [synthmata](https://synthmata.github.io/volca-fm/) online editor (requires [additional hardware](https://github.com/probonopd/MiniDexed/wiki/Hardware#usb-midi-device)) +- [x] Voices can be edited over MIDI, e.g., using the [synthmata](https://synthmata.github.io/volca-fm/) online editor (requires [additional hardware](https://github.com/probonopd/MiniDexed/wiki/Hardware#usb-midi-devices)) ## Introduction Video about this project by [Floyd Steinberg](https://www.youtube.com/watch?v=Z3t94ceMHJo): -[![](https://i.ytimg.com/vi/Z3t94ceMHJo/sddefault.jpg)](https://www.youtube.com/watch?v=Z3t94ceMHJo) +[![YouTube Video about MiniDexed (Floyd Steinberg)](https://i.ytimg.com/vi/Z3t94ceMHJo/sddefault.jpg)](https://www.youtube.com/watch?v=Z3t94ceMHJo) ## System Requirements -* Raspberry Pi 1, 2, 3, 4, or 400 (Zero and Zero 2 can be used but need HDMI or a supported i2s DAC for audio out). On Raspberry Pi 1 and on Raspberry Pi Zero there will be severely limited functionality (only one tone generator instead of 8) -* A [PCM5102A or PCM5122 based DAC](https://github.com/probonopd/MiniDexed/wiki/Hardware#i2c-dac), HDMI display or [audio extractor](https://github.com/probonopd/MiniDexed/wiki/Hardware#hdmi-to-audio) for good sound quality. If you don't have this, you can use the headphone jack on the Raspberry Pi but on anything but the Raspberry 4 the sound quality will be seriously limited -* Optionally (but highly recommended), an [LCDC1602 Display](https://www.berrybase.de/en/sensors-modules/displays/alphanumeric-displays/alphanumerisches-lcd-16x2-gr-252-n/gelb) (with or without i2c "backpack" board) and a [KY-040 rotary encoder](https://www.berrybase.de/en/components/passive-components/potentiometer/rotary-encoder/drehregler/rotary-encoder-mit-breakoutboard-ohne-gewinde-und-mutter) +- Raspberry Pi 1, 2, 3, 4, or 400 (Zero and Zero 2 can be used but need HDMI or a supported i2s DAC for audio out). On Raspberry Pi 1 and on Raspberry Pi Zero there will be severely limited functionality (only one tone generator instead of 8) +- A [PCM5102A or PCM5122 based DAC](https://github.com/probonopd/MiniDexed/wiki/Hardware#i2s-dac), HDMI display or [audio extractor](https://github.com/probonopd/MiniDexed/wiki/Hardware#hdmi-to-audio) for good sound quality. If you don't have this, you can use the headphone jack on the Raspberry Pi but on anything but the Raspberry 4 the sound quality will be seriously limited +- Optionally (but highly recommended), an [LCDC1602 Display](https://www.berrybase.de/en/sensors-modules/displays/alphanumeric-displays/alphanumerisches-lcd-16x2-gr-252-n/gelb) (with or without i2c "backpack" board) and a [KY-040 rotary encoder](https://www.berrybase.de/en/components/passive-components/potentiometer/rotary-encoder/drehregler/rotary-encoder-mit-breakoutboard-ohne-gewinde-und-mutter) ## Usage -* In the case of Raspberry Pi 4, Update the firmware and bootloader to the latest version (not doing this may cause USB reliability issues) -* Download from [GitHub Releases](../../releases) -* Unzip -* Put the files into the root directory of a FAT32 formatted partition on SD/microSD card (Note for small SD cards which are no longer sold: If less than 65525 clusters, you may need to format as FAT16.) -* Put SD/microSD card into Raspberry Pi 1, 2, 3 or 4, or 400 (Zero and Zero 2 can be used but need HDMI or a supported i2c DAC for audio out) -* Attach headphones to the headphone jack using `SoundDevice=pwm` in `minidexed.ini` (default) (poor audio quality) -* Alternatively, attach a PCM5102A or PCM5122 based DAC and select i2c sound output using `SoundDevice=i2s` in `minidexed.ini` (best audio quality) -* Alternatively, attach a HDMI display with sound and select HDMI sound output using `SoundDevice=hdmi` in `minidexed.ini` (this may introduce slight latency) -* Attach a MIDI keyboard via USB (alternatively you can build a circuit that allows you to attach a "traditional" MIDI keyboard using a DIN connector, or use a DIN-MIDI-to-USB adapter) -* If you are using a LCDC1602 with an i2c "backpack" board, then you need to set `LCDI2CAddress=0x27` (or another address your i2c "backpack" board is set to) in `minidexed.ini` -* Boot -* Start playing -* If the system seems to become unresponsive after a few seconds, remove `usbspeed=full` from `cmdline.txt` and repeat ([details](https://github.com/probonopd/MiniDexed/issues/39)) -* Optionally, put voices in `.syx` files onto the SD card (e.g., using `getsysex.sh`) -* See the Wiki for [Menu](https://github.com/probonopd/MiniDexed/wiki/Menu) operation -* For voice programming, use any DX series editor (using MIDI sysex), including Dexed -* For library management, use the dedicated [MiniDexedLibrarian](https://github.com/BobanSpasic/MiniDexedLibrarian) software -* If something is unclear or does not work, don't hesitate to [ask](https://github.com/probonopd/MiniDexed/discussions/)! +- In the case of Raspberry Pi 4, Update the firmware and bootloader to the latest version (not doing this may cause USB reliability issues) +- Download from [GitHub Releases](../../releases) +- Unzip +- Put the files into the root directory of a FAT32 formatted partition on SD/microSD card (Note for small SD cards which are no longer sold: If less than 65525 clusters, you may need to format as FAT16.) +- Put SD/microSD card into Raspberry Pi 1, 2, 3 or 4, or 400 (Zero and Zero 2 can be used but need HDMI or a supported i2c DAC for audio out) +- Attach headphones to the headphone jack using `SoundDevice=pwm` in `minidexed.ini` (default) (poor audio quality) +- Alternatively, attach a PCM5102A or PCM5122 based DAC and select i2c sound output using `SoundDevice=i2s` in `minidexed.ini` (best audio quality) +- Alternatively, attach a HDMI display with sound and select HDMI sound output using `SoundDevice=hdmi` in `minidexed.ini` (this may introduce slight latency) +- Attach a MIDI keyboard via USB (alternatively you can build a circuit that allows you to attach a "traditional" MIDI keyboard using a DIN connector, or use a DIN-MIDI-to-USB adapter) +- If you are using a LCDC1602 with an i2c "backpack" board, then you need to set `LCDI2CAddress=0x27` (or another address your i2c "backpack" board is set to) in `minidexed.ini` +- Boot +- Start playing +- If the system seems to become unresponsive after a few seconds, remove `usbspeed=full` from `cmdline.txt` and repeat ([details](https://github.com/probonopd/MiniDexed/issues/39)) +- Optionally, put voices in `.syx` files onto the SD card (e.g., using `getsysex.sh`) +- See the Wiki for [Menu](https://github.com/probonopd/MiniDexed/wiki/Menu) operation +- For voice programming, use any DX series editor (using MIDI sysex), including Dexed +- For library management, use the dedicated [MiniDexedLibrarian](https://github.com/BobanSpasic/MiniDexedLibrarian) software +- If something is unclear or does not work, don't hesitate to [ask](https://github.com/probonopd/MiniDexed/discussions/)! ## Pinout All devices on Raspberry Pi GPIOs are **optional**. -![](https://user-images.githubusercontent.com/2480569/166105580-da11481c-8fc7-4375-8ab1-3031ab5c6ad0.png) +![Raspberry Pi Pinout/GPIO Diagram](https://user-images.githubusercontent.com/2480569/166105580-da11481c-8fc7-4375-8ab1-3031ab5c6ad0.png) -Please the the [wiki](https://github.com/probonopd/MiniDexed/wiki) for more information. +Please see the [wiki](https://github.com/probonopd/MiniDexed/wiki) for more information. ## Downloading @@ -71,24 +71,24 @@ Please see the [wiki](https://github.com/probonopd/MiniDexed/wiki/Development#bu ## Contributing -This project lives from the contributions of skilled C++ developers, testers, writers, etc. Please see https://github.com/probonopd/MiniDexed/issues. +This project lives from the contributions of skilled C++ developers, testers, writers, etc. Please see . ## Discussions -We are happy to hear from you. Please join the discussions on https://github.com/probonopd/MiniDexed/discussions. +We are happy to hear from you. Please join the discussions on . ## Documentation -Project documentation is at https://github.com/probonopd/MiniDexed/wiki. +Project documentation is at . ## Acknowledgements This project stands on the shoulders of giants. Special thanks to: -* [raphlinus](https://github.com/raphlinus) for the [MSFA](https://github.com/google/music-synthesizer-for-android) sound engine -* [asb2m10](https://github.com/asb2m10/dexed) for the [Dexed](https://github.com/asb2m10/dexed) software -* [dcoredump](https://github.com/dcoredump) for https://codeberg.org/dcoredump/Synth_Dexed, a port of Dexed for embedded systems -* [rsta2](https://github.com/rsta2) for https://github.com/rsta2/circle, the library to run code on bare metal Raspberry Pi (without a Linux kernel or operating system) and for the bulk of the MiniDexed code -* [smuehlst](https://github.com/smuehlst) for https://github.com/smuehlst/circle-stdlib, a version with Standard C and C++ library support -* [Banana71](https://github.com/Banana71) for the sound design of the [Soundplantage](https://github.com/Banana71/Soundplantage) performances shipped with MiniDexed -* [BobanSpasic](https://github.com/BobanSpasic) for the [MiniDexedLibrarian](https://github.com/BobanSpasic/MiniDexedLibrarian) software, [MiniDexed performance converter](https://github.com/BobanSpasic/MDX_PerfConv) and [collection of performances for MiniDexed](https://github.com/BobanSpasic/MDX_Vault) +- [raphlinus](https://github.com/raphlinus) for the [MSFA](https://github.com/google/music-synthesizer-for-android) sound engine +- [asb2m10](https://github.com/asb2m10/dexed) for the [Dexed](https://github.com/asb2m10/dexed) software +- [dcoredump](https://github.com/dcoredump) for [Synth Dexed](https://codeberg.org/dcoredump/Synth_Dexed), a port of Dexed for embedded systems +- [rsta2](https://github.com/rsta2) for [Circle](https://github.com/rsta2/circle), the library to run code on bare metal Raspberry Pi (without a Linux kernel or operating system) and for the bulk of the MiniDexed code +- [smuehlst](https://github.com/smuehlst) for [circle-stdlib](https://github.com/smuehlst/circle-stdlib), a version with Standard C and C++ library support +- [Banana71](https://github.com/Banana71) for the sound design of the [Soundplantage](https://github.com/Banana71/Soundplantage) performances shipped with MiniDexed +- [BobanSpasic](https://github.com/BobanSpasic) for the [MiniDexedLibrarian](https://github.com/BobanSpasic/MiniDexedLibrarian) software, [MiniDexed performance converter](https://github.com/BobanSpasic/MDX_PerfConv) and [collection of performances for MiniDexed](https://github.com/BobanSpasic/MDX_Vault) From cc7b1a5ac2df0222f5d2f54805a2756c58cbd64f Mon Sep 17 00:00:00 2001 From: probonopd Date: Mon, 11 Mar 2024 21:57:52 +0100 Subject: [PATCH 03/38] diyelectromusic [ci skip] --- README.md | 1 + 1 file changed, 1 insertion(+) diff --git a/README.md b/README.md index a0e6ae5..ead3e58 100644 --- a/README.md +++ b/README.md @@ -92,3 +92,4 @@ This project stands on the shoulders of giants. Special thanks to: - [smuehlst](https://github.com/smuehlst) for [circle-stdlib](https://github.com/smuehlst/circle-stdlib), a version with Standard C and C++ library support - [Banana71](https://github.com/Banana71) for the sound design of the [Soundplantage](https://github.com/Banana71/Soundplantage) performances shipped with MiniDexed - [BobanSpasic](https://github.com/BobanSpasic) for the [MiniDexedLibrarian](https://github.com/BobanSpasic/MiniDexedLibrarian) software, [MiniDexed performance converter](https://github.com/BobanSpasic/MDX_PerfConv) and [collection of performances for MiniDexed](https://github.com/BobanSpasic/MDX_Vault) +- [diyelectromusic](https://github.com/diyelectromusic/) for many [contributions](https://github.com/probonopd/MiniDexed/commits?author=diyelectromusic) From 544aaff5b8a970dbebbfdb74a4302630cdddd53c Mon Sep 17 00:00:00 2001 From: Kevin <68612569+diyelectromusic@users.noreply.github.com> Date: Mon, 25 Mar 2024 13:32:23 +0000 Subject: [PATCH 04/38] Fix for Issue #628 - MIDI Button Channel 16 treated as OMNI by mistake. (#629) --- src/userinterface.cpp | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/userinterface.cpp b/src/userinterface.cpp index a2d78f1..241605e 100644 --- a/src/userinterface.cpp +++ b/src/userinterface.cpp @@ -390,10 +390,10 @@ void CUserInterface::UISetMIDIButtonChannel (unsigned uCh) m_nMIDIButtonCh = CMIDIDevice::Disabled; LOGNOTE("MIDI Button channel not set"); } - else if (uCh < CMIDIDevice::Channels) + else if (uCh <= CMIDIDevice::Channels) { m_nMIDIButtonCh = uCh - 1; - LOGNOTE("MIDI Button channel set to: %d", m_nMIDIButtonCh); + LOGNOTE("MIDI Button channel set to: %d", m_nMIDIButtonCh+1); } else { From 082445b6304ffb4cd59897fab20c836ad602284d Mon Sep 17 00:00:00 2001 From: probonopd Date: Mon, 25 Mar 2024 13:51:22 +0000 Subject: [PATCH 05/38] Default MIDIAutoVoiceDumpOnPC to 0 (#614) Fixes #611 --- src/config.cpp | 2 +- src/config.h | 2 +- src/minidexed.ini | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/src/config.cpp b/src/config.cpp index 526d8ad..bb65840 100644 --- a/src/config.cpp +++ b/src/config.cpp @@ -82,7 +82,7 @@ void CConfig::Load (void) m_bMIDIRXProgramChange = m_Properties.GetNumber ("MIDIRXProgramChange", 1) != 0; m_bIgnoreAllNotesOff = m_Properties.GetNumber ("IgnoreAllNotesOff", 0) != 0; - m_bMIDIAutoVoiceDumpOnPC = m_Properties.GetNumber ("MIDIAutoVoiceDumpOnPC", 1) != 0; + m_bMIDIAutoVoiceDumpOnPC = m_Properties.GetNumber ("MIDIAutoVoiceDumpOnPC", 0) != 0; m_bHeaderlessSysExVoices = m_Properties.GetNumber ("HeaderlessSysExVoices", 0) != 0; m_bExpandPCAcrossBanks = m_Properties.GetNumber ("ExpandPCAcrossBanks", 1) != 0; diff --git a/src/config.h b/src/config.h index f83c177..55c038e 100644 --- a/src/config.h +++ b/src/config.h @@ -80,7 +80,7 @@ public: const char *GetMIDIThruOut (void) const; // "" if not specified bool GetMIDIRXProgramChange (void) const; // true if not specified bool GetIgnoreAllNotesOff (void) const; - bool GetMIDIAutoVoiceDumpOnPC (void) const; // true if not specified + bool GetMIDIAutoVoiceDumpOnPC (void) const; // false if not specified bool GetHeaderlessSysExVoices (void) const; // false if not specified bool GetExpandPCAcrossBanks (void) const; // true if not specified diff --git a/src/minidexed.ini b/src/minidexed.ini index cd5ea1a..84d76ac 100644 --- a/src/minidexed.ini +++ b/src/minidexed.ini @@ -17,7 +17,7 @@ EngineType=1 MIDIBaudRate=31250 #MIDIThru=umidi1,ttyS1 IgnoreAllNotesOff=0 -MIDIAutoVoiceDumpOnPC=1 +MIDIAutoVoiceDumpOnPC=0 HeaderlessSysExVoices=0 # Program Change enable # 0 = Ignore all Program Change messages. From 349100d65ad59e73b091031ea011b0f73242890b Mon Sep 17 00:00:00 2001 From: probonopd Date: Mon, 25 Mar 2024 16:12:44 +0100 Subject: [PATCH 06/38] Demo songs [ci skip] --- README.md | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/README.md b/README.md index ead3e58..5c5aca4 100644 --- a/README.md +++ b/README.md @@ -4,6 +4,10 @@ MiniDexed is a FM synthesizer closely modeled on the famous DX7 by a well-known Japanese manufacturer running on a bare metal Raspberry Pi (without a Linux kernel or operating system). On Raspberry Pi 2 and larger, it can run 8 tone generators, not unlike the TX816/TX802 (8 DX7 instances without the keyboard in one box). [Featured by HACKADAY](https://hackaday.com/2022/04/19/bare-metal-gives-this-pi-some-classic-synths/), [Adafruit](https://blog.adafruit.com/2022/04/25/free-yamaha-dx7-synth-emulator-on-a-raspberry-pi/), and [Synth Geekery](https://www.youtube.com/watch?v=TDSy5nnm0jA). +## Demo songs + +Listen to some examples made with MiniDexed by Banana71 [here](https://soundcloud.com/soundplantage/sets/minidexed2). + ## Features - [x] Uses [Synth_Dexed](https://codeberg.org/dcoredump/Synth_Dexed) with [circle-stdlib](https://github.com/smuehlst/circle-stdlib) From 75eb763055739e4a5e5625e273ed4ab6252f0a24 Mon Sep 17 00:00:00 2001 From: Kevin <68612569+diyelectromusic@users.noreply.github.com> Date: Wed, 17 Apr 2024 20:21:18 +0100 Subject: [PATCH 07/38] Very early initial support for RPi 5 based on dev branch of circle. (#638) * Very early initial support for RPi 5 based on dev branch of circle. * Added parameter to choose UART option on all RPi versions to be GP14/15 * Build for Raspberry Pi 5 * Update README.md for RPi 5 --------- Co-authored-by: probonopd --- .github/workflows/build.yml | 6 ++++++ README.md | 3 ++- src/config.txt | 6 +++++- src/kernel.cpp | 4 ++++ src/minidexed.cpp | 8 ++++++++ src/serialmididevice.cpp | 6 +++++- src/usbminidexedmidigadget.h | 18 ++++++++++++++++++ submod.sh | 4 ++-- 8 files changed, 50 insertions(+), 5 deletions(-) diff --git a/.github/workflows/build.yml b/.github/workflows/build.yml index 56ef1fd..8df5911 100644 --- a/.github/workflows/build.yml +++ b/.github/workflows/build.yml @@ -27,6 +27,12 @@ jobs: wget -q https://developer.arm.com/-/media/Files/downloads/gnu-a/10.3-2021.07/binrel/gcc-arm-10.3-2021.07-x86_64-arm-none-eabi.tar.xz tar xf *-arm-none-eabi.tar.xz mkdir -p kernels + - name: Build for Raspberry Pi 5 + run: | + set -ex + export PATH=$(readlink -f ./gcc-*aarch64-none*/bin/):$PATH + RPI=5 bash -ex build.sh + cp ./src/kernel*.img ./kernels/ - name: Build for Raspberry Pi 4 run: | set -ex diff --git a/README.md b/README.md index 5c5aca4..e1807bf 100644 --- a/README.md +++ b/README.md @@ -32,7 +32,8 @@ Video about this project by [Floyd Steinberg](https://www.youtube.com/watch?v=Z3 ## System Requirements -- Raspberry Pi 1, 2, 3, 4, or 400 (Zero and Zero 2 can be used but need HDMI or a supported i2s DAC for audio out). On Raspberry Pi 1 and on Raspberry Pi Zero there will be severely limited functionality (only one tone generator instead of 8) +- Raspberry Pi 1, 2, 3, 4, or 400. Raspberry Pi Zero and Zero 2 can be used but need HDMI or a supported i2s DAC for audio out. On Raspberry Pi 1 and on Raspberry Pi Zero there will be severely limited functionality (only one tone generator instead of 8) +- Raspberry Pi 5 can be used but currently support is experimental; HDMI sound and USB Gadget mode are not available yet - A [PCM5102A or PCM5122 based DAC](https://github.com/probonopd/MiniDexed/wiki/Hardware#i2s-dac), HDMI display or [audio extractor](https://github.com/probonopd/MiniDexed/wiki/Hardware#hdmi-to-audio) for good sound quality. If you don't have this, you can use the headphone jack on the Raspberry Pi but on anything but the Raspberry 4 the sound quality will be seriously limited - Optionally (but highly recommended), an [LCDC1602 Display](https://www.berrybase.de/en/sensors-modules/displays/alphanumeric-displays/alphanumerisches-lcd-16x2-gr-252-n/gelb) (with or without i2c "backpack" board) and a [KY-040 rotary encoder](https://www.berrybase.de/en/components/passive-components/potentiometer/rotary-encoder/drehregler/rotary-encoder-mit-breakoutboard-ohne-gewinde-und-mutter) diff --git a/src/config.txt b/src/config.txt index 7957b5c..c10dfa1 100644 --- a/src/config.txt +++ b/src/config.txt @@ -10,7 +10,7 @@ gpu_mem=16 disable_overscan=0 # -# Use 64-bit for RPi 3, 4, 400 and Zero 2, and 32-bit for all other models +# Use 64-bit for RPi 3, 4, 400, 5 and Zero 2, and 32-bit for all other models # [pi3] @@ -24,3 +24,7 @@ kernel=kernel8-rpi4.img # Zero 2 W [pi02] arm_64bit=1 + +[pi5] +arm_64bit=1 +kernel=kernel_2712.img diff --git a/src/kernel.cpp b/src/kernel.cpp index c06c386..3ff835c 100644 --- a/src/kernel.cpp +++ b/src/kernel.cpp @@ -68,8 +68,12 @@ bool CKernel::Initialize (void) if (m_Config.GetUSBGadgetMode()) { +#if RASPPI==5 +#warning No support for USB Gadget Mode on RPI 5 yet +#else // Run the USB stack in USB Gadget (device) mode m_pUSB = new CUSBMiniDexedMIDIGadget (&mInterrupt); +#endif } else { diff --git a/src/minidexed.cpp b/src/minidexed.cpp index 968320a..e2576b5 100644 --- a/src/minidexed.cpp +++ b/src/minidexed.cpp @@ -103,7 +103,11 @@ CMiniDexed::CMiniDexed (CConfig *pConfig, CInterruptSystem *pInterrupt, if (pConfig->GetUSBGadgetMode()) { +#if RASPPI==5 + LOGNOTE ("USB Gadget (Device) Mode NOT supported on RPI 5"); +#else LOGNOTE ("USB In Gadget (Device) Mode"); +#endif } else { @@ -128,6 +132,9 @@ CMiniDexed::CMiniDexed (CConfig *pConfig, CInterruptSystem *pInterrupt, } else if (strcmp (pDeviceName, "hdmi") == 0) { +#if RASPPI==5 + LOGNOTE ("HDMI mode NOT supported on RPI 5."); +#else LOGNOTE ("HDMI mode"); m_pSoundDevice = new CHDMISoundBaseDevice (pInterrupt, pConfig->GetSampleRate (), @@ -136,6 +143,7 @@ CMiniDexed::CMiniDexed (CConfig *pConfig, CInterruptSystem *pInterrupt, // The channels are swapped by default in the HDMI sound driver. // TODO: Remove this line, when this has been fixed in the driver. m_bChannelsSwapped = !m_bChannelsSwapped; +#endif } else { diff --git a/src/serialmididevice.cpp b/src/serialmididevice.cpp index 883fd4d..b37a498 100644 --- a/src/serialmididevice.cpp +++ b/src/serialmididevice.cpp @@ -28,11 +28,15 @@ LOGMODULE("serialmididevice"); +// There are several UART options - see circle/include/serial.h +// 0 corresponds to GP14/GP15 on all RPi versions. +#define SERIAL_MIDI_DEVICE 0 + CSerialMIDIDevice::CSerialMIDIDevice (CMiniDexed *pSynthesizer, CInterruptSystem *pInterrupt, CConfig *pConfig, CUserInterface *pUI) : CMIDIDevice (pSynthesizer, pConfig, pUI), m_pConfig (pConfig), - m_Serial (pInterrupt, TRUE), + m_Serial (pInterrupt, TRUE, SERIAL_MIDI_DEVICE), m_nSerialState (0), m_nSysEx (0), m_SendBuffer (&m_Serial) diff --git a/src/usbminidexedmidigadget.h b/src/usbminidexedmidigadget.h index bbb5f44..8513549 100644 --- a/src/usbminidexedmidigadget.h +++ b/src/usbminidexedmidigadget.h @@ -22,6 +22,23 @@ #ifndef _usbminidexedmidigadget_h #define _usbminidexedmidigadget_h +#if RASPPI==5 +#include +#include + +#warning No support for USB Gadget Mode on RPI 5 yet +class CUSBMiniDexedMIDIGadget +{ +public: + CUSBMiniDexedMIDIGadget (CInterruptSystem *pInterruptSystem) + { + } + + ~CUSBMiniDexedMIDIGadget (void) + { + } +}; +#else #include #include #include @@ -82,5 +99,6 @@ protected: return CUSBMIDIGadget::GetDescriptor(wValue, wIndex, pLength); } }; +#endif #endif diff --git a/submod.sh b/submod.sh index 2e24228..d72d19c 100755 --- a/submod.sh +++ b/submod.sh @@ -6,13 +6,13 @@ git submodule update --init --recursive # # Use fixed master branch of circle-stdlib then re-update cd circle-stdlib/ -git checkout 695ab4a +git checkout 3bd135d git submodule update --init --recursive cd - # # Optional update submodules explicitly cd circle-stdlib/libs/circle -git checkout fe09c4b +git checkout 4b3e06f cd - cd circle-stdlib/libs/circle-newlib #git checkout develop From e2dc897c8a7402c49bd23b6a2d3428b77f2bf4c1 Mon Sep 17 00:00:00 2001 From: Kevin <68612569+diyelectromusic@users.noreply.github.com> Date: Wed, 24 Apr 2024 21:13:17 +0100 Subject: [PATCH 08/38] Update README.md (#643) [ci skip] * Update README.md Added recommendation to use an older or cheaper Pi (not a Pi 5) for a first build. * Update README.md --------- Co-authored-by: probonopd --- README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.md b/README.md index e1807bf..dcb9d84 100644 --- a/README.md +++ b/README.md @@ -33,7 +33,7 @@ Video about this project by [Floyd Steinberg](https://www.youtube.com/watch?v=Z3 ## System Requirements - Raspberry Pi 1, 2, 3, 4, or 400. Raspberry Pi Zero and Zero 2 can be used but need HDMI or a supported i2s DAC for audio out. On Raspberry Pi 1 and on Raspberry Pi Zero there will be severely limited functionality (only one tone generator instead of 8) -- Raspberry Pi 5 can be used but currently support is experimental; HDMI sound and USB Gadget mode are not available yet +- Raspberry Pi 5 can be used but currently support is experimental: HDMI sound and USB Gadget mode are not available yet, and it is not clear if there are implications for cooling from running MiniDexed. Also, MiniDexed is currently not taking advantage of the higher processing power of the Raspberry Pi 5 yet. *Hence, you may consider using one of the less expensive, older Raspberry Pi boards for your first build.* - A [PCM5102A or PCM5122 based DAC](https://github.com/probonopd/MiniDexed/wiki/Hardware#i2s-dac), HDMI display or [audio extractor](https://github.com/probonopd/MiniDexed/wiki/Hardware#hdmi-to-audio) for good sound quality. If you don't have this, you can use the headphone jack on the Raspberry Pi but on anything but the Raspberry 4 the sound quality will be seriously limited - Optionally (but highly recommended), an [LCDC1602 Display](https://www.berrybase.de/en/sensors-modules/displays/alphanumeric-displays/alphanumerisches-lcd-16x2-gr-252-n/gelb) (with or without i2c "backpack" board) and a [KY-040 rotary encoder](https://www.berrybase.de/en/components/passive-components/potentiometer/rotary-encoder/drehregler/rotary-encoder-mit-breakoutboard-ohne-gewinde-und-mutter) From 4fa9e167b27f59490f86d74f628cc7e624c1c4fd Mon Sep 17 00:00:00 2001 From: Kevin <68612569+diyelectromusic@users.noreply.github.com> Date: Thu, 25 Apr 2024 18:30:02 +0100 Subject: [PATCH 09/38] Initial implementation of USB MIDI SysEx message handling. (#634) * Initial implementation of USB MIDI SysEx message handling. * Remove unused parameter from USB message handling. --- src/mididevice.h | 3 ++ src/midikeyboard.cpp | 67 +++++++++++++++++++++++++++++++++++++--- src/midikeyboard.h | 7 +++++ src/serialmididevice.cpp | 4 +-- src/serialmididevice.h | 3 -- 5 files changed, 75 insertions(+), 9 deletions(-) diff --git a/src/mididevice.h b/src/mididevice.h index 44be25a..0fc027c 100644 --- a/src/mididevice.h +++ b/src/mididevice.h @@ -30,6 +30,9 @@ #include #include "userinterface.h" +#define MAX_DX7_SYSEX_LENGTH 4104 +#define MAX_MIDI_MESSAGE MAX_DX7_SYSEX_LENGTH + class CMiniDexed; class CMIDIDevice diff --git a/src/midikeyboard.cpp b/src/midikeyboard.cpp index 169f165..2eae9d8 100644 --- a/src/midikeyboard.cpp +++ b/src/midikeyboard.cpp @@ -37,6 +37,7 @@ TMIDIPacketHandler * const CMIDIKeyboard::s_pMIDIPacketHandler[MaxInstances] = CMIDIKeyboard::CMIDIKeyboard (CMiniDexed *pSynthesizer, CConfig *pConfig, CUserInterface *pUI, unsigned nInstance) : CMIDIDevice (pSynthesizer, pConfig, pUI), + m_nSysExIdx (0), m_nInstance (nInstance), m_pMIDIDevice (0) { @@ -100,28 +101,86 @@ void CMIDIKeyboard::Send (const u8 *pMessage, size_t nLength, unsigned nCable) m_SendQueue.push (Entry); } +// Most packets will be passed straight onto the main MIDI message handler +// but SysEx messages are multiple USB packets and so will need building up +// before parsing. +void CMIDIKeyboard::USBMIDIMessageHandler (u8 *pPacket, unsigned nLength, unsigned nCable) +{ + if ((pPacket[0] == 0xF0) && (m_nSysExIdx == 0)) + { + // Start of SysEx message + //printf("SysEx Start Idx=%d, (%d)\n", m_nSysExIdx, nLength); + for (unsigned i=0; i= USB_SYSEX_BUFFER_SIZE) { + // Run out of space, so reset and ignore rest of the message + m_nSysExIdx = 0; + break; + } + else if (pPacket[i] == 0xF7) { + // End of SysEx message + m_SysEx[m_nSysExIdx++] = pPacket[i]; + //printf ("SysEx End Idx=%d\n", m_nSysExIdx); + MIDIMessageHandler (m_SysEx, m_nSysExIdx, nCable); + // Reset ready for next time + m_nSysExIdx = 0; + } + else if ((pPacket[i] & 0x80) != 0) { + // Received another command, so reset processing as something has gone wrong + //printf ("SysEx Reset\n"); + m_nSysExIdx = 0; + break; + } + else + { + // Store the byte + m_SysEx[m_nSysExIdx++] = pPacket[i]; + } + } + } + else + { + // Assume it is a standard message + MIDIMessageHandler (pPacket, nLength, nCable); + } +} + void CMIDIKeyboard::MIDIPacketHandler0 (unsigned nCable, u8 *pPacket, unsigned nLength) { assert (s_pThis[0] != 0); - s_pThis[0]->MIDIMessageHandler (pPacket, nLength, nCable); + s_pThis[0]->USBMIDIMessageHandler (pPacket, nLength, nCable); } void CMIDIKeyboard::MIDIPacketHandler1 (unsigned nCable, u8 *pPacket, unsigned nLength) { assert (s_pThis[1] != 0); - s_pThis[1]->MIDIMessageHandler (pPacket, nLength, nCable); + s_pThis[1]->USBMIDIMessageHandler (pPacket, nLength, nCable); } void CMIDIKeyboard::MIDIPacketHandler2 (unsigned nCable, u8 *pPacket, unsigned nLength) { assert (s_pThis[2] != 0); - s_pThis[2]->MIDIMessageHandler (pPacket, nLength, nCable); + s_pThis[2]->USBMIDIMessageHandler (pPacket, nLength, nCable); } void CMIDIKeyboard::MIDIPacketHandler3 (unsigned nCable, u8 *pPacket, unsigned nLength) { assert (s_pThis[3] != 0); - s_pThis[3]->MIDIMessageHandler (pPacket, nLength, nCable); + s_pThis[3]->USBMIDIMessageHandler (pPacket, nLength, nCable); } void CMIDIKeyboard::DeviceRemovedHandler (CDevice *pDevice, void *pContext) diff --git a/src/midikeyboard.h b/src/midikeyboard.h index 0868f9c..047fa52 100644 --- a/src/midikeyboard.h +++ b/src/midikeyboard.h @@ -31,6 +31,8 @@ #include #include +#define USB_SYSEX_BUFFER_SIZE (MAX_DX7_SYSEX_LENGTH+128) // Allow a bit spare to handle unexpected SysEx messages + class CMiniDexed; class CMIDIKeyboard : public CMIDIDevice @@ -53,6 +55,8 @@ private: static void MIDIPacketHandler3 (unsigned nCable, u8 *pPacket, unsigned nLength); static void DeviceRemovedHandler (CDevice *pDevice, void *pContext); + + void USBMIDIMessageHandler (u8 *pPacket, unsigned nLength, unsigned nCable); private: struct TSendQueueEntry @@ -61,6 +65,8 @@ private: size_t nLength; unsigned nCable; }; + uint8_t m_SysEx[USB_SYSEX_BUFFER_SIZE]; + unsigned m_nSysExIdx; private: unsigned m_nInstance; @@ -73,6 +79,7 @@ private: static CMIDIKeyboard *s_pThis[MaxInstances]; static TMIDIPacketHandler * const s_pMIDIPacketHandler[MaxInstances]; + }; #endif diff --git a/src/serialmididevice.cpp b/src/serialmididevice.cpp index b37a498..3fe01e8 100644 --- a/src/serialmididevice.cpp +++ b/src/serialmididevice.cpp @@ -75,7 +75,7 @@ void CSerialMIDIDevice::Process (void) return; } - if (m_pConfig->GetMIDIDumpEnabled ()) +/* if (m_pConfig->GetMIDIDumpEnabled ()) { printf("Incoming MIDI data:"); for (uint16_t i = 0; i < nResult; i++) @@ -85,7 +85,7 @@ void CSerialMIDIDevice::Process (void) printf(" 0x%02x",Buffer[i]); } printf("\n"); - } + }*/ // Process MIDI messages // See: https://www.midi.org/specifications/item/table-1-summary-of-midi-message diff --git a/src/serialmididevice.h b/src/serialmididevice.h index 1a5b465..fdf61b2 100644 --- a/src/serialmididevice.h +++ b/src/serialmididevice.h @@ -30,9 +30,6 @@ #include #include -#define MAX_DX7_SYSEX_LENGTH 4104 -#define MAX_MIDI_MESSAGE MAX_DX7_SYSEX_LENGTH - class CMiniDexed; class CSerialMIDIDevice : public CMIDIDevice From d08280bc7005499773df5c27fe9f0d53cda1ebab Mon Sep 17 00:00:00 2001 From: Kevin <68612569+diyelectromusic@users.noreply.github.com> Date: Thu, 9 May 2024 18:58:46 +0100 Subject: [PATCH 10/38] Support for ST7789 based (SPI) displays (#652) * Initial build with basic structure to support st7789device once added to circle. * Implementation of ST7789 display - requires updated circle with new ST7789 character driver. * Added more details to minidexed.ini about ST7789 and SPI options. * Update to develop branch of circle that now supports st7789 character mode display. * Minor formatting fixes to tidy up. * Allow setting of more advanced SPI parameters: mode and clock. * Update to allow for font size as an option. --- src/config.cpp | 73 +++++++++++++++++++++++++++++++++++++++++++ src/config.h | 36 ++++++++++++++++++++- src/kernel.cpp | 29 ++++++++++++++++- src/kernel.h | 2 ++ src/minidexed.cpp | 4 +-- src/minidexed.h | 3 +- src/minidexed.ini | 21 +++++++++++++ src/userinterface.cpp | 69 +++++++++++++++++++++++++++++++++++----- src/userinterface.h | 7 ++++- submod.sh | 2 +- 10 files changed, 232 insertions(+), 14 deletions(-) diff --git a/src/config.cpp b/src/config.cpp index bb65840..d456c58 100644 --- a/src/config.cpp +++ b/src/config.cpp @@ -102,6 +102,20 @@ void CConfig::Load (void) m_bSSD1306LCDRotate = m_Properties.GetNumber ("SSD1306LCDRotate", 0) != 0; m_bSSD1306LCDMirror = m_Properties.GetNumber ("SSD1306LCDMirror", 0) != 0; + m_nSPIBus = m_Properties.GetNumber ("SPIBus", SPI_INACTIVE); // Disabled by default + m_nSPIMode = m_Properties.GetNumber ("SPIMode", SPI_DEF_MODE); + m_nSPIClockKHz = m_Properties.GetNumber ("SPIClockKHz", SPI_DEF_CLOCK); + + m_bST7789Enabled = m_Properties.GetNumber ("ST7789Enabled", 0) != 0; + m_nST7789Data = m_Properties.GetNumber ("ST7789Data", 0); + m_nST7789Select = m_Properties.GetNumber ("ST7789Select", 0); + m_nST7789Reset = m_Properties.GetNumber ("ST7789Reset", 0); // optional + m_nST7789Backlight = m_Properties.GetNumber ("ST7789Backlight", 0); // optional + m_nST7789Width = m_Properties.GetNumber ("ST7789Width", 240); + m_nST7789Height = m_Properties.GetNumber ("ST7789Height", 240); + m_nST7789Rotation = m_Properties.GetNumber ("ST7789Rotation", 0); + m_bST7789SmallFont = m_Properties.GetNumber ("ST7789SmallFont", 0) != 0; + m_nLCDColumns = m_Properties.GetNumber ("LCDColumns", 16); m_nLCDRows = m_Properties.GetNumber ("LCDRows", 2); @@ -299,6 +313,65 @@ bool CConfig::GetSSD1306LCDMirror (void) const return m_bSSD1306LCDMirror; } +unsigned CConfig::GetSPIBus (void) const +{ + return m_nSPIBus; +} + +unsigned CConfig::GetSPIMode (void) const +{ + return m_nSPIMode; +} + +unsigned CConfig::GetSPIClockKHz (void) const +{ + return m_nSPIClockKHz; +} + +bool CConfig::GetST7789Enabled (void) const +{ + return m_bST7789Enabled; +} + +unsigned CConfig::GetST7789Data (void) const +{ + return m_nST7789Data; +} + +unsigned CConfig::GetST7789Select (void) const +{ + return m_nST7789Select; +} + +unsigned CConfig::GetST7789Reset (void) const +{ + return m_nST7789Reset; +} + +unsigned CConfig::GetST7789Backlight (void) const +{ + return m_nST7789Backlight; +} + +unsigned CConfig::GetST7789Width (void) const +{ + return m_nST7789Width; +} + +unsigned CConfig::GetST7789Height (void) const +{ + return m_nST7789Height; +} + +unsigned CConfig::GetST7789Rotation (void) const +{ + return m_nST7789Rotation; +} + +bool CConfig::GetST7789SmallFont (void) const +{ + return m_bST7789SmallFont; +} unsigned CConfig::GetLCDColumns (void) const { return m_nLCDColumns; diff --git a/src/config.h b/src/config.h index 55c038e..71bb5ad 100644 --- a/src/config.h +++ b/src/config.h @@ -28,6 +28,10 @@ #include #include +#define SPI_INACTIVE 255 +#define SPI_DEF_CLOCK 15000 // kHz +#define SPI_DEF_MODE 0 // Default mode (0,1,2,3) + class CConfig // Configuration for MiniDexed { public: @@ -103,6 +107,22 @@ public: bool GetSSD1306LCDRotate (void) const; bool GetSSD1306LCDMirror (void) const; + // SPI support + unsigned GetSPIBus (void) const; + unsigned GetSPIMode (void) const; + unsigned GetSPIClockKHz (void) const; + + // ST7789 LCD + bool GetST7789Enabled (void) const; + unsigned GetST7789Data (void) const; + unsigned GetST7789Select (void) const; + unsigned GetST7789Reset (void) const; + unsigned GetST7789Backlight (void) const; + unsigned GetST7789Width (void) const; + unsigned GetST7789Height (void) const; + unsigned GetST7789Rotation (void) const; + bool GetST7789SmallFont (void) const; + unsigned GetLCDColumns (void) const; unsigned GetLCDRows (void) const; @@ -204,7 +224,21 @@ private: unsigned m_nSSD1306LCDHeight; bool m_bSSD1306LCDRotate; bool m_bSSD1306LCDMirror; - + + unsigned m_nSPIBus; + unsigned m_nSPIMode; + unsigned m_nSPIClockKHz; + + bool m_bST7789Enabled; + unsigned m_nST7789Data; + unsigned m_nST7789Select; + unsigned m_nST7789Reset; + unsigned m_nST7789Backlight; + unsigned m_nST7789Width; + unsigned m_nST7789Height; + unsigned m_nST7789Rotation; + unsigned m_bST7789SmallFont; + unsigned m_nLCDColumns; unsigned m_nLCDRows; diff --git a/src/kernel.cpp b/src/kernel.cpp index 3ff835c..33bc5f8 100644 --- a/src/kernel.cpp +++ b/src/kernel.cpp @@ -33,6 +33,7 @@ CKernel::CKernel (void) m_Config (&mFileSystem), m_GPIOManager (&mInterrupt), m_I2CMaster (CMachineInfo::Get ()->GetDevice (DeviceI2CMaster), TRUE), + m_pSPIMaster (nullptr), m_pDexed (0) { s_pThis = this; @@ -66,6 +67,32 @@ bool CKernel::Initialize (void) m_Config.Load (); + unsigned nSPIMaster = m_Config.GetSPIBus(); + unsigned nSPIMode = m_Config.GetSPIMode(); + unsigned long nSPIClock = 1000 * m_Config.GetSPIClockKHz(); +#if RASPPI<4 + // By default older RPI versions use SPI 0. + // It is possible to build circle to support SPI 1 for + // devices that use the 40-pin header, but that isn't + // enabled at present... + if (nSPIMaster == 0) +#else + // RPI 4+ has several possible SPI Bus Configurations. + // As mentioned above, SPI 1 is not built by default. + // See circle/include/circle/spimaster.h + if (nSPIMaster == 0 || nSPIMaster == 3 || nSPIMaster == 4 || nSPIMaster == 5 || nSPIMaster == 6) +#endif + { + unsigned nCPHA = (nSPIMode & 1) ? 1 : 0; + unsigned nCPOL = (nSPIMode & 2) ? 1 : 0; + m_pSPIMaster = new CSPIMaster (nSPIClock, nCPOL, nCPHA, nSPIMaster); + if (!m_pSPIMaster->Initialize()) + { + delete (m_pSPIMaster); + m_pSPIMaster = nullptr; + } + } + if (m_Config.GetUSBGadgetMode()) { #if RASPPI==5 @@ -86,7 +113,7 @@ bool CKernel::Initialize (void) return FALSE; } - m_pDexed = new CMiniDexed (&m_Config, &mInterrupt, &m_GPIOManager, &m_I2CMaster, + m_pDexed = new CMiniDexed (&m_Config, &mInterrupt, &m_GPIOManager, &m_I2CMaster, m_pSPIMaster, &mFileSystem); assert (m_pDexed); diff --git a/src/kernel.h b/src/kernel.h index 7d2f346..efe5f4f 100644 --- a/src/kernel.h +++ b/src/kernel.h @@ -24,6 +24,7 @@ #include #include #include +#include #include #include "config.h" #include "minidexed.h" @@ -54,6 +55,7 @@ private: CCPUThrottle m_CPUThrottle; CGPIOManager m_GPIOManager; CI2CMaster m_I2CMaster; + CSPIMaster *m_pSPIMaster; CMiniDexed *m_pDexed; CUSBController *m_pUSB; diff --git a/src/minidexed.cpp b/src/minidexed.cpp index e2576b5..9547baf 100644 --- a/src/minidexed.cpp +++ b/src/minidexed.cpp @@ -31,13 +31,13 @@ LOGMODULE ("minidexed"); CMiniDexed::CMiniDexed (CConfig *pConfig, CInterruptSystem *pInterrupt, - CGPIOManager *pGPIOManager, CI2CMaster *pI2CMaster, FATFS *pFileSystem) + CGPIOManager *pGPIOManager, CI2CMaster *pI2CMaster, CSPIMaster *pSPIMaster, FATFS *pFileSystem) : #ifdef ARM_ALLOW_MULTI_CORE CMultiCoreSupport (CMemorySystem::Get ()), #endif m_pConfig (pConfig), - m_UI (this, pGPIOManager, pI2CMaster, pConfig), + m_UI (this, pGPIOManager, pI2CMaster, pSPIMaster, pConfig), m_PerformanceConfig (pFileSystem), m_PCKeyboard (this, pConfig, &m_UI), m_SerialMIDI (this, pInterrupt, pConfig, &m_UI), diff --git a/src/minidexed.h b/src/minidexed.h index 407d5db..8ca74c8 100644 --- a/src/minidexed.h +++ b/src/minidexed.h @@ -36,6 +36,7 @@ #include #include #include +#include #include #include #include @@ -51,7 +52,7 @@ class CMiniDexed { public: CMiniDexed (CConfig *pConfig, CInterruptSystem *pInterrupt, - CGPIOManager *pGPIOManager, CI2CMaster *pI2CMaster, FATFS *pFileSystem); + CGPIOManager *pGPIOManager, CI2CMaster *pI2CMaster, CSPIMaster *pSPIMaster, FATFS *pFileSystem); bool Initialize (void); diff --git a/src/minidexed.ini b/src/minidexed.ini index 84d76ac..de9f150 100644 --- a/src/minidexed.ini +++ b/src/minidexed.ini @@ -55,6 +55,27 @@ SSD1306LCDHeight=32 SSD1306LCDRotate=0 SSD1306LCDMirror=0 +# ST7789 LCD +# SPIBus=0 for any RPi (GPIO 10,11,8,7). +# Note: Leave blank (default) if no SPI device required. +# Select=0|1 for CE0 or CE1 +# Data = GPIO pin number +# Optional: Reset, Backlight = GPIO pin numbers +# Rotation=0,90,180,270 +# SmallFont=0 (default), 1 +# +# For a 240 wide display set LCDColumns=15 with LCDRows=2 +SPIBus= +ST7789Enabled=0 +ST7789Data= +ST7789Select= +ST7789Reset= +ST7789Backlight= +ST7789Width=240 +ST7789Height=240 +ST7789Rotation=0 +ST7789SmallFont=0 + # Default is 16x2 display (e.g. HD44780) LCDColumns=16 LCDRows=2 diff --git a/src/userinterface.cpp b/src/userinterface.cpp index 241605e..9e7c112 100644 --- a/src/userinterface.cpp +++ b/src/userinterface.cpp @@ -27,10 +27,11 @@ LOGMODULE ("ui"); -CUserInterface::CUserInterface (CMiniDexed *pMiniDexed, CGPIOManager *pGPIOManager, CI2CMaster *pI2CMaster, CConfig *pConfig) +CUserInterface::CUserInterface (CMiniDexed *pMiniDexed, CGPIOManager *pGPIOManager, CI2CMaster *pI2CMaster, CSPIMaster *pSPIMaster, CConfig *pConfig) : m_pMiniDexed (pMiniDexed), m_pGPIOManager (pGPIOManager), m_pI2CMaster (pI2CMaster), + m_pSPIMaster (pSPIMaster), m_pConfig (pConfig), m_pLCD (0), m_pLCDBuffered (0), @@ -57,17 +58,69 @@ bool CUserInterface::Initialize (void) { unsigned i2caddr = m_pConfig->GetLCDI2CAddress (); unsigned ssd1306addr = m_pConfig->GetSSD1306LCDI2CAddress (); + bool st7789 = m_pConfig->GetST7789Enabled (); if (ssd1306addr != 0) { m_pSSD1306 = new CSSD1306Device (m_pConfig->GetSSD1306LCDWidth (), m_pConfig->GetSSD1306LCDHeight (), m_pI2CMaster, ssd1306addr, m_pConfig->GetSSD1306LCDRotate (), m_pConfig->GetSSD1306LCDMirror ()); - LOGDBG ("LCD: SSD1306"); if (!m_pSSD1306->Initialize ()) { + LOGDBG("LCD: SSD1306 initialization failed"); return false; } + LOGDBG ("LCD: SSD1306"); m_pLCD = m_pSSD1306; - } else if (i2caddr == 0) + } + else if (st7789) + { + if (m_pSPIMaster == nullptr) + { + LOGDBG("LCD: ST7789 Enabled but SPI Initialisation Failed"); + return false; + } + + unsigned long nSPIClock = 1000 * m_pConfig->GetSPIClockKHz(); + unsigned nSPIMode = m_pConfig->GetSPIMode(); + unsigned nCPHA = (nSPIMode & 1) ? 1 : 0; + unsigned nCPOL = (nSPIMode & 2) ? 1 : 0; + LOGDBG("SPI: CPOL=%u; CPHA=%u; CLK=%u",nCPOL,nCPHA,nSPIClock); + m_pST7789Display = new CST7789Display (m_pSPIMaster, + m_pConfig->GetST7789Data(), + m_pConfig->GetST7789Reset(), + m_pConfig->GetST7789Backlight(), + m_pConfig->GetST7789Width(), + m_pConfig->GetST7789Height(), + nCPOL, nCPHA, nSPIClock, + m_pConfig->GetST7789Select()); + if (m_pST7789Display->Initialize()) + { + m_pST7789Display->SetRotation (m_pConfig->GetST7789Rotation()); + bool bLargeFont = !(m_pConfig->GetST7789SmallFont()); + m_pST7789 = new CST7789Device (m_pSPIMaster, m_pST7789Display, m_pConfig->GetLCDColumns (), m_pConfig->GetLCDRows (), bLargeFont, bLargeFont); + if (m_pST7789->Initialize()) + { + LOGDBG ("LCD: ST7789"); + m_pLCD = m_pST7789; + } + else + { + LOGDBG ("LCD: Failed to initalize ST7789 character device"); + delete (m_pST7789); + delete (m_pST7789Display); + m_pST7789 = nullptr; + m_pST7789Display = nullptr; + return false; + } + } + else + { + LOGDBG ("LCD: Failed to initialize ST7789 display"); + delete (m_pST7789Display); + m_pST7789Display = nullptr; + return false; + } + } + else if (i2caddr == 0) { m_pHD44780 = new CHD44780Device (m_pConfig->GetLCDColumns (), m_pConfig->GetLCDRows (), m_pConfig->GetLCDPinData4 (), @@ -77,22 +130,24 @@ bool CUserInterface::Initialize (void) m_pConfig->GetLCDPinEnable (), m_pConfig->GetLCDPinRegisterSelect (), m_pConfig->GetLCDPinReadWrite ()); - LOGDBG ("LCD: HD44780"); if (!m_pHD44780->Initialize ()) { + LOGDBG("LCD: HD44780 initialization failed"); return false; } + LOGDBG ("LCD: HD44780"); m_pLCD = m_pHD44780; } else { m_pHD44780 = new CHD44780Device (m_pI2CMaster, i2caddr, m_pConfig->GetLCDColumns (), m_pConfig->GetLCDRows ()); - LOGDBG ("LCD: HD44780 I2C"); if (!m_pHD44780->Initialize ()) { + LOGDBG("LCD: HD44780 (I2C) initialization failed"); return false; } + LOGDBG ("LCD: HD44780 I2C"); m_pLCD = m_pHD44780; } assert (m_pLCD); @@ -217,7 +272,7 @@ void CUserInterface::DisplayWrite (const char *pMenu, const char *pParam, const CString Value (" "); if (bArrowDown) { - Value = "\x7F"; // arrow left character + Value = "<"; // arrow left character } Value.Append (pValue); @@ -232,7 +287,7 @@ void CUserInterface::DisplayWrite (const char *pMenu, const char *pParam, const } } - Value.Append ("\x7E"); // arrow right character + Value.Append (">"); // arrow right character } Msg.Append (Value); diff --git a/src/userinterface.h b/src/userinterface.h index 5de2846..a8026db 100644 --- a/src/userinterface.h +++ b/src/userinterface.h @@ -26,16 +26,18 @@ #include #include #include +#include #include #include #include +#include class CMiniDexed; class CUserInterface { public: - CUserInterface (CMiniDexed *pMiniDexed, CGPIOManager *pGPIOManager, CI2CMaster *pI2CMaster, CConfig *pConfig); + CUserInterface (CMiniDexed *pMiniDexed, CGPIOManager *pGPIOManager, CI2CMaster *pI2CMaster, CSPIMaster *pSPIMaster, CConfig *pConfig); ~CUserInterface (void); bool Initialize (void); @@ -68,11 +70,14 @@ private: CMiniDexed *m_pMiniDexed; CGPIOManager *m_pGPIOManager; CI2CMaster *m_pI2CMaster; + CSPIMaster *m_pSPIMaster; CConfig *m_pConfig; CCharDevice *m_pLCD; CHD44780Device *m_pHD44780; CSSD1306Device *m_pSSD1306; + CST7789Display *m_pST7789Display; + CST7789Device *m_pST7789; CWriteBufferDevice *m_pLCDBuffered; CUIButtons *m_pUIButtons; diff --git a/submod.sh b/submod.sh index d72d19c..c13dca3 100755 --- a/submod.sh +++ b/submod.sh @@ -12,7 +12,7 @@ cd - # # Optional update submodules explicitly cd circle-stdlib/libs/circle -git checkout 4b3e06f +git checkout 4155f43 cd - cd circle-stdlib/libs/circle-newlib #git checkout develop From afa72d21aa81a9876b3f9e50f3bd6f4cd56cdac0 Mon Sep 17 00:00:00 2001 From: probonopd Date: Thu, 30 May 2024 18:43:48 +0200 Subject: [PATCH 11/38] Attempt to fix #646 (#649) * Attempt to fix #646 https://github.com/probonopd/MiniDexed/issues/648 * SetVoiceName() suggested by @diyelectromusic --- src/minidexed.cpp | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/src/minidexed.cpp b/src/minidexed.cpp index 9547baf..02d8fd6 100644 --- a/src/minidexed.cpp +++ b/src/minidexed.cpp @@ -1767,8 +1767,9 @@ void CMiniDexed::SetVoiceName (std::string VoiceName, unsigned nTG) { assert (nTG < CConfig::ToneGenerators); assert (m_pTG[nTG]); - char Name[10]; + char Name[11]; strncpy(Name, VoiceName.c_str(),10); + Name[10] = '\0'; m_pTG[nTG]->getName (Name); } From 98b5274cf3889577d7a402abc5ddce529e233350 Mon Sep 17 00:00:00 2001 From: Kevin <68612569+diyelectromusic@users.noreply.github.com> Date: Sat, 29 Jun 2024 22:44:52 +0100 Subject: [PATCH 12/38] Add support for 8 channel I2S mono audio output on RPi 5 (#657) * First implementation of using four PCM5102 I2S sound devices for 8-channel mono sound output on a Raspberry Pi 5. Requires latest develop branch of circle. * Update to required develop branch of circle * Adjusted default chunk sizes to correctly support number of channels. * Update queue size as per Rene's suggestion. --------- Co-authored-by: probonopd --- src/config.cpp | 18 +++- src/config.h | 2 + src/minidexed.cpp | 222 ++++++++++++++++++++++++++++++---------------- src/minidexed.h | 1 + submod.sh | 2 +- 5 files changed, 167 insertions(+), 78 deletions(-) diff --git a/src/config.cpp b/src/config.cpp index d456c58..ea6bde3 100644 --- a/src/config.cpp +++ b/src/config.cpp @@ -41,15 +41,22 @@ void CConfig::Load (void) m_SoundDevice = m_Properties.GetString ("SoundDevice", "pwm"); m_nSampleRate = m_Properties.GetNumber ("SampleRate", 48000); + m_bQuadDAC8Chan = m_Properties.GetNumber ("QuadDAC8Chan", 0) != 0; + if (m_SoundDevice == "hdmi") { + m_nChunkSize = m_Properties.GetNumber ("ChunkSize", 384*6); + } + else + { #ifdef ARM_ALLOW_MULTI_CORE - m_nChunkSize = m_Properties.GetNumber ("ChunkSize", m_SoundDevice == "hdmi" ? 384*6 : 256); + m_nChunkSize = m_Properties.GetNumber ("ChunkSize", m_bQuadDAC8Chan ? 1024 : 256); // 128 per channel #else - m_nChunkSize = m_Properties.GetNumber ("ChunkSize", m_SoundDevice == "hdmi" ? 384*6 : 1024); + m_nChunkSize = m_Properties.GetNumber ("ChunkSize", 1024); #endif + } m_nDACI2CAddress = m_Properties.GetNumber ("DACI2CAddress", 0); m_bChannelsSwapped = m_Properties.GetNumber ("ChannelsSwapped", 0) != 0; - unsigned newEngineType = m_Properties.GetNumber ("EngineType", 1); + unsigned newEngineType = m_Properties.GetNumber ("EngineType", 1); if (newEngineType == 2) { m_EngineType = MKI; } else if (newEngineType == 3) { @@ -243,6 +250,11 @@ bool CConfig::GetExpandPCAcrossBanks (void) const return m_bExpandPCAcrossBanks; } +bool CConfig::GetQuadDAC8Chan (void) const +{ + return m_bQuadDAC8Chan; +} + bool CConfig::GetLCDEnabled (void) const { return m_bLCDEnabled; diff --git a/src/config.h b/src/config.h index 71bb5ad..5944dd2 100644 --- a/src/config.h +++ b/src/config.h @@ -87,6 +87,7 @@ public: bool GetMIDIAutoVoiceDumpOnPC (void) const; // false if not specified bool GetHeaderlessSysExVoices (void) const; // false if not specified bool GetExpandPCAcrossBanks (void) const; // true if not specified + bool GetQuadDAC8Chan (void) const; // false if not specified // HD44780 LCD // GPIO pin numbers are chip numbers, not header positions @@ -208,6 +209,7 @@ private: bool m_bMIDIAutoVoiceDumpOnPC; bool m_bHeaderlessSysExVoices; bool m_bExpandPCAcrossBanks; + bool m_bQuadDAC8Chan; bool m_bLCDEnabled; unsigned m_nLCDPinEnable; diff --git a/src/minidexed.cpp b/src/minidexed.cpp index 02d8fd6..429fddc 100644 --- a/src/minidexed.cpp +++ b/src/minidexed.cpp @@ -42,6 +42,7 @@ CMiniDexed::CMiniDexed (CConfig *pConfig, CInterruptSystem *pInterrupt, m_PCKeyboard (this, pConfig, &m_UI), m_SerialMIDI (this, pInterrupt, pConfig, &m_UI), m_bUseSerial (false), + m_bQuadDAC8Chan (false), m_pSoundDevice (0), m_bChannelsSwapped (pConfig->GetChannelsSwapped ()), #ifdef ARM_ALLOW_MULTI_CORE @@ -125,10 +126,26 @@ CMiniDexed::CMiniDexed (CConfig *pConfig, CInterruptSystem *pInterrupt, if (strcmp (pDeviceName, "i2s") == 0) { LOGNOTE ("I2S mode"); - - m_pSoundDevice = new CI2SSoundBaseDevice (pInterrupt, pConfig->GetSampleRate (), - pConfig->GetChunkSize (), false, - pI2CMaster, pConfig->GetDACI2CAddress ()); +#if RASPPI==5 + // Quad DAC 8-channel mono only an option for RPI 5 + m_bQuadDAC8Chan = pConfig->GetQuadDAC8Chan (); +#endif + if (m_bQuadDAC8Chan) { + LOGNOTE ("Configured for Quad DAC 8-channel Mono audio"); + m_pSoundDevice = new CI2SSoundBaseDevice (pInterrupt, pConfig->GetSampleRate (), + pConfig->GetChunkSize (), false, + pI2CMaster, pConfig->GetDACI2CAddress (), + CI2SSoundBaseDevice::DeviceModeTXOnly, + 8); // 8 channels - L+R x4 across 4 I2S lanes + } + else + { + m_pSoundDevice = new CI2SSoundBaseDevice (pInterrupt, pConfig->GetSampleRate (), + pConfig->GetChunkSize (), false, + pI2CMaster, pConfig->GetDACI2CAddress (), + CI2SSoundBaseDevice::DeviceModeTXOnly, + 2); // 2 channels - L+R + } } else if (strcmp (pDeviceName, "hdmi") == 0) { @@ -251,18 +268,30 @@ bool CMiniDexed::Initialize (void) } // setup and start the sound device - if (!m_pSoundDevice->AllocateQueueFrames (m_pConfig->GetChunkSize ())) + int Channels = 1; // 16-bit Mono +#ifdef ARM_ALLOW_MULTI_CORE + if (m_bQuadDAC8Chan) + { + Channels = 8; // 16-bit 8-channel mono + } + else + { + Channels = 2; // 16-bit Stereo + } +#endif + // Need 2 x ChunkSize / Channel queue frames as the audio driver uses + // two DMA channels each of ChunkSize and one single single frame + // contains a sample for each of all the channels. + // + // See discussion here: https://github.com/rsta2/circle/discussions/453 + if (!m_pSoundDevice->AllocateQueueFrames (2 * m_pConfig->GetChunkSize () / Channels)) { LOGERR ("Cannot allocate sound queue"); return false; } -#ifndef ARM_ALLOW_MULTI_CORE - m_pSoundDevice->SetWriteFormat (SoundFormatSigned16, 1); // 16-bit Mono -#else - m_pSoundDevice->SetWriteFormat (SoundFormatSigned16, 2); // 16-bit Stereo -#endif + m_pSoundDevice->SetWriteFormat (SoundFormatSigned16, Channels); m_nQueueSizeFrames = m_pSoundDevice->GetQueueSizeFrames (); @@ -1128,85 +1157,130 @@ void CMiniDexed::ProcessSound (void) assert (CConfig::ToneGenerators == 8); - uint8_t indexL=0, indexR=1; - - // BEGIN TG mixing - float32_t tmp_float[nFrames*2]; - int16_t tmp_int[nFrames*2]; + if (m_bQuadDAC8Chan) { + // No mixing is performed by MiniDexed, sound is output in 8 channels. + // Note: one TG per audio channel; output=mono; no processing. + const int Channels = 8; // One TG per channel + float32_t tmp_float[nFrames*Channels]; + int16_t tmp_int[nFrames*Channels]; - if(nMasterVolume > 0.0) - { - for (uint8_t i = 0; i < CConfig::ToneGenerators; i++) + if(nMasterVolume > 0.0) { - tg_mixer->doAddMix(i,m_OutputLevel[i]); - reverb_send_mixer->doAddMix(i,m_OutputLevel[i]); + // Convert dual float array (8 chan) to single int16 array (8 chan) + for(uint16_t i=0; i0.0 && nMasterVolume <1.0) + { + tmp_float[(i*Channels)+tg]=m_OutputLevel[tg][i] * nMasterVolume; + } + else if(nMasterVolume == 1.0) + { + tmp_float[(i*Channels)+tg]=m_OutputLevel[tg][i]; + } + } + } + arm_float_to_q15(tmp_float,tmp_int,nFrames*Channels); } - // END TG mixing - - // BEGIN create SampleBuffer for holding audio data - float32_t SampleBuffer[2][nFrames]; - // END create SampleBuffer for holding audio data - - // get the mix of all TGs - tg_mixer->getMix(SampleBuffer[indexL], SampleBuffer[indexR]); - - // BEGIN adding reverb - if (m_nParameter[ParameterReverbEnable]) + else { - float32_t ReverbBuffer[2][nFrames]; - float32_t ReverbSendBuffer[2][nFrames]; - - arm_fill_f32(0.0f, ReverbBuffer[indexL], nFrames); - arm_fill_f32(0.0f, ReverbBuffer[indexR], nFrames); - arm_fill_f32(0.0f, ReverbSendBuffer[indexR], nFrames); - arm_fill_f32(0.0f, ReverbSendBuffer[indexL], nFrames); - - m_ReverbSpinLock.Acquire (); - - reverb_send_mixer->getMix(ReverbSendBuffer[indexL], ReverbSendBuffer[indexR]); - reverb->doReverb(ReverbSendBuffer[indexL],ReverbSendBuffer[indexR],ReverbBuffer[indexL], ReverbBuffer[indexR],nFrames); - - // scale down and add left reverb buffer by reverb level - arm_scale_f32(ReverbBuffer[indexL], reverb->get_level(), ReverbBuffer[indexL], nFrames); - arm_add_f32(SampleBuffer[indexL], ReverbBuffer[indexL], SampleBuffer[indexL], nFrames); - // scale down and add right reverb buffer by reverb level - arm_scale_f32(ReverbBuffer[indexR], reverb->get_level(), ReverbBuffer[indexR], nFrames); - arm_add_f32(SampleBuffer[indexR], ReverbBuffer[indexR], SampleBuffer[indexR], nFrames); - - m_ReverbSpinLock.Release (); + arm_fill_q15(0, tmp_int, nFrames*Channels); } - // END adding reverb - - // swap stereo channels if needed prior to writing back out - if (m_bChannelsSwapped) + + if (m_pSoundDevice->Write (tmp_int, sizeof(tmp_int)) != (int) sizeof(tmp_int)) { - indexL=1; - indexR=0; + LOGERR ("Sound data dropped"); } + } + else + { + // Mix everything down to stereo + uint8_t indexL=0, indexR=1; + + // BEGIN TG mixing + float32_t tmp_float[nFrames*2]; + int16_t tmp_int[nFrames*2]; - // Convert dual float array (left, right) to single int16 array (left/right) - for(uint16_t i=0; i 0.0) { - if(nMasterVolume >0.0 && nMasterVolume <1.0) + for (uint8_t i = 0; i < CConfig::ToneGenerators; i++) { - tmp_float[i*2]=SampleBuffer[indexL][i] * nMasterVolume; - tmp_float[(i*2)+1]=SampleBuffer[indexR][i] * nMasterVolume; + tg_mixer->doAddMix(i,m_OutputLevel[i]); + reverb_send_mixer->doAddMix(i,m_OutputLevel[i]); } - else if(nMasterVolume == 1.0) + // END TG mixing + + // BEGIN create SampleBuffer for holding audio data + float32_t SampleBuffer[2][nFrames]; + // END create SampleBuffer for holding audio data + + // get the mix of all TGs + tg_mixer->getMix(SampleBuffer[indexL], SampleBuffer[indexR]); + + // BEGIN adding reverb + if (m_nParameter[ParameterReverbEnable]) { - tmp_float[i*2]=SampleBuffer[indexL][i]; - tmp_float[(i*2)+1]=SampleBuffer[indexR][i]; + float32_t ReverbBuffer[2][nFrames]; + float32_t ReverbSendBuffer[2][nFrames]; + + arm_fill_f32(0.0f, ReverbBuffer[indexL], nFrames); + arm_fill_f32(0.0f, ReverbBuffer[indexR], nFrames); + arm_fill_f32(0.0f, ReverbSendBuffer[indexR], nFrames); + arm_fill_f32(0.0f, ReverbSendBuffer[indexL], nFrames); + + m_ReverbSpinLock.Acquire (); + + reverb_send_mixer->getMix(ReverbSendBuffer[indexL], ReverbSendBuffer[indexR]); + reverb->doReverb(ReverbSendBuffer[indexL],ReverbSendBuffer[indexR],ReverbBuffer[indexL], ReverbBuffer[indexR],nFrames); + + // scale down and add left reverb buffer by reverb level + arm_scale_f32(ReverbBuffer[indexL], reverb->get_level(), ReverbBuffer[indexL], nFrames); + arm_add_f32(SampleBuffer[indexL], ReverbBuffer[indexL], SampleBuffer[indexL], nFrames); + // scale down and add right reverb buffer by reverb level + arm_scale_f32(ReverbBuffer[indexR], reverb->get_level(), ReverbBuffer[indexR], nFrames); + arm_add_f32(SampleBuffer[indexR], ReverbBuffer[indexR], SampleBuffer[indexR], nFrames); + + m_ReverbSpinLock.Release (); } + // END adding reverb + + // swap stereo channels if needed prior to writing back out + if (m_bChannelsSwapped) + { + indexL=1; + indexR=0; + } + + // Convert dual float array (left, right) to single int16 array (left/right) + for(uint16_t i=0; i0.0 && nMasterVolume <1.0) + { + tmp_float[i*2]=SampleBuffer[indexL][i] * nMasterVolume; + tmp_float[(i*2)+1]=SampleBuffer[indexR][i] * nMasterVolume; + } + else if(nMasterVolume == 1.0) + { + tmp_float[i*2]=SampleBuffer[indexL][i]; + tmp_float[(i*2)+1]=SampleBuffer[indexR][i]; + } + } + arm_float_to_q15(tmp_float,tmp_int,nFrames*2); + } + else + { + arm_fill_q15(0, tmp_int, nFrames * 2); } - arm_float_to_q15(tmp_float,tmp_int,nFrames*2); - } - else - arm_fill_q15(0, tmp_int, nFrames * 2); - if (m_pSoundDevice->Write (tmp_int, sizeof(tmp_int)) != (int) sizeof(tmp_int)) - { - LOGERR ("Sound data dropped"); - } + if (m_pSoundDevice->Write (tmp_int, sizeof(tmp_int)) != (int) sizeof(tmp_int)) + { + LOGERR ("Sound data dropped"); + } + } // End of Stereo mixing if (m_bProfileEnabled) { diff --git a/src/minidexed.h b/src/minidexed.h index 8ca74c8..6e5e012 100644 --- a/src/minidexed.h +++ b/src/minidexed.h @@ -298,6 +298,7 @@ private: CPCKeyboard m_PCKeyboard; CSerialMIDIDevice m_SerialMIDI; bool m_bUseSerial; + bool m_bQuadDAC8Chan; CSoundBaseDevice *m_pSoundDevice; bool m_bChannelsSwapped; diff --git a/submod.sh b/submod.sh index c13dca3..f9524a3 100755 --- a/submod.sh +++ b/submod.sh @@ -12,7 +12,7 @@ cd - # # Optional update submodules explicitly cd circle-stdlib/libs/circle -git checkout 4155f43 +git checkout fff3764 cd - cd circle-stdlib/libs/circle-newlib #git checkout develop From 75e82caf72c2154c9c3c909835d09d005bbc24bc Mon Sep 17 00:00:00 2001 From: probonopd Date: Fri, 5 Jul 2024 18:58:01 +0200 Subject: [PATCH 13/38] Try USBGadgetPin (#670) * Try USBGadgetPin * CGPIOPin usbGadgetPin(usbGadgetPinNumber, GPIOModeInputPullUp); * Remove extraneous line --- src/config.cpp | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/src/config.cpp b/src/config.cpp index ea6bde3..ad55d13 100644 --- a/src/config.cpp +++ b/src/config.cpp @@ -22,6 +22,7 @@ // #include "config.h" #include "../Synth_Dexed/src/dexed.h" +#include CConfig::CConfig (FATFS *pFileSystem) : m_Properties ("minidexed.ini", pFileSystem) @@ -37,6 +38,13 @@ void CConfig::Load (void) m_Properties.Load (); m_bUSBGadgetMode = m_Properties.GetNumber ("USBGadget", 0) != 0; + unsigned usbGadgetPinNumber = m_Properties.GetNumber ("USBGadgetPin", 26); // Default to GPIO pin 26 if not specified + CGPIOPin usbGadgetPin(usbGadgetPinNumber, GPIOModeInputPullUp); + + if (usbGadgetPin.Read() == 0) // If the pin is pulled down + { + m_bUSBGadgetMode = true; + } m_SoundDevice = m_Properties.GetString ("SoundDevice", "pwm"); From 24a309ebb44fb4831ff5629c2cbf020008375d63 Mon Sep 17 00:00:00 2001 From: probonopd Date: Thu, 11 Jul 2024 19:18:36 +0200 Subject: [PATCH 14/38] Add artifact links to PR and issues --- .github/workflows/pr-comment.yml | 78 ++++++++------------------------ 1 file changed, 20 insertions(+), 58 deletions(-) diff --git a/.github/workflows/pr-comment.yml b/.github/workflows/pr-comment.yml index c5c8ebb..179dc7b 100644 --- a/.github/workflows/pr-comment.yml +++ b/.github/workflows/pr-comment.yml @@ -1,64 +1,26 @@ -# https://nightly.link/ -# https://github.com/oprypin/nightly.link/blob/master/.github/workflows/pr-comment.yml +# https://github.com/subsurface/subsurface/blob/master/.github/workflows/artifact-links.yml + +name: Add artifact links to pull request -name: Comment on pull request on: workflow_run: - workflows: ['Build'] + workflows: ["Build"] types: [completed] -jobs: - pr_comment: - if: github.event.workflow_run.event == 'pull_request' && github.event.workflow_run.conclusion == 'success' - runs-on: ubuntu-latest - steps: - - uses: actions/github-script@v6 - with: - # This snippet is public-domain, taken from - # https://github.com/oprypin/nightly.link/blob/master/.github/workflows/pr-comment.yml - script: | - async function upsertComment(owner, repo, issue_number, purpose, body) { - const {data: comments} = await github.rest.issues.listComments( - {owner, repo, issue_number}); - - const marker = ``; - body = marker + "\n" + body; - - const existing = comments.filter((c) => c.body.includes(marker)); - if (existing.length > 0) { - const last = existing[existing.length - 1]; - core.info(`Updating comment ${last.id}`); - await github.rest.issues.updateComment({ - owner, repo, - body, - comment_id: last.id, - }); - } else { - core.info(`Creating a comment in issue / PR #${issue_number}`); - await github.rest.issues.createComment({issue_number, body, owner, repo}); - } - } - const {owner, repo} = context.repo; - const run_id = ${{github.event.workflow_run.id}}; - - const pull_requests = ${{ toJSON(github.event.workflow_run.pull_requests) }}; - if (!pull_requests.length) { - return core.error("This workflow doesn't match any pull requests!"); - } - - const artifacts = await github.paginate( - github.rest.actions.listWorkflowRunArtifacts, {owner, repo, run_id}); - if (!artifacts.length) { - return core.error(`No artifacts found`); - } - let body = `Download the artifacts for this pull request:\n`; - for (const art of artifacts) { - body += `\n* [${art.name}.zip](https://nightly.link/${owner}/${repo}/actions/artifacts/${art.id}.zip)`; - } - - core.info("Review thread message body:", body); +jobs: + artifacts-url-comments: + name: Add artifact links to PR and issues + runs-on: ubuntu-22.04 - for (const pr of pull_requests) { - await upsertComment(owner, repo, pr.number, - "nightly-link", body); - } + steps: + - name: Add artifact links to PR and issues + if: github.event.workflow_run.event == 'pull_request' + uses: tonyhallett/artifacts-url-comments@v1.1.0 + env: + GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} + with: + prefix: "Build for testing:" + suffix: "Use at your own risk._" + format: name + addTo: pull + errorNoArtifacts: false From 6664e63fe517c26e0759c354dbeea67f128bfecd Mon Sep 17 00:00:00 2001 From: probonopd Date: Thu, 11 Jul 2024 20:04:30 +0200 Subject: [PATCH 15/38] Fix typo [ci skip] --- .github/workflows/pr-comment.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/pr-comment.yml b/.github/workflows/pr-comment.yml index 179dc7b..74b0c0f 100644 --- a/.github/workflows/pr-comment.yml +++ b/.github/workflows/pr-comment.yml @@ -20,7 +20,7 @@ jobs: GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} with: prefix: "Build for testing:" - suffix: "Use at your own risk._" + suffix: "Use at your own risk." format: name addTo: pull errorNoArtifacts: false From 3b033db4afd54cef2f7af2a22ec5957422393722 Mon Sep 17 00:00:00 2001 From: Kevin <68612569+diyelectromusic@users.noreply.github.com> Date: Sat, 13 Jul 2024 17:21:09 +0100 Subject: [PATCH 16/38] Implement USBGadgetPin and appropriate boot messages. (#674) --- src/config.cpp | 27 ++++++++++++++++++--------- src/config.h | 7 ++++++- src/kernel.cpp | 28 ++++++++++++++++++++++++++-- src/minidexed.cpp | 32 +++++++++++++++++++++++++++++--- 4 files changed, 79 insertions(+), 15 deletions(-) diff --git a/src/config.cpp b/src/config.cpp index ad55d13..884d7cc 100644 --- a/src/config.cpp +++ b/src/config.cpp @@ -22,7 +22,6 @@ // #include "config.h" #include "../Synth_Dexed/src/dexed.h" -#include CConfig::CConfig (FATFS *pFileSystem) : m_Properties ("minidexed.ini", pFileSystem) @@ -37,14 +36,9 @@ void CConfig::Load (void) { m_Properties.Load (); - m_bUSBGadgetMode = m_Properties.GetNumber ("USBGadget", 0) != 0; - unsigned usbGadgetPinNumber = m_Properties.GetNumber ("USBGadgetPin", 26); // Default to GPIO pin 26 if not specified - CGPIOPin usbGadgetPin(usbGadgetPinNumber, GPIOModeInputPullUp); - - if (usbGadgetPin.Read() == 0) // If the pin is pulled down - { - m_bUSBGadgetMode = true; - } + m_bUSBGadget = m_Properties.GetNumber ("USBGadget", 0) != 0; + m_nUSBGadgetPin = m_Properties.GetNumber ("USBGadgetPin", 0); // Default OFF + SetUSBGadgetMode(m_bUSBGadget); // Might get overriden later by USBGadgetPin state m_SoundDevice = m_Properties.GetString ("SoundDevice", "pwm"); @@ -183,11 +177,26 @@ void CConfig::Load (void) m_bPerformanceSelectChannel = m_Properties.GetNumber ("PerformanceSelectChannel", 0); } +bool CConfig::GetUSBGadget (void) const +{ + return m_bUSBGadget; +} + +unsigned CConfig::GetUSBGadgetPin (void) const +{ + return m_nUSBGadgetPin; +} + bool CConfig::GetUSBGadgetMode (void) const { return m_bUSBGadgetMode; } +void CConfig::SetUSBGadgetMode (bool USBGadgetMode) +{ + m_bUSBGadgetMode = USBGadgetMode; +} + const char *CConfig::GetSoundDevice (void) const { return m_SoundDevice.c_str (); diff --git a/src/config.h b/src/config.h index 5944dd2..7497c21 100644 --- a/src/config.h +++ b/src/config.h @@ -68,7 +68,10 @@ public: void Load (void); // USB Mode - bool GetUSBGadgetMode (void) const; // true if in USB gadget mode + bool GetUSBGadget (void) const; + unsigned GetUSBGadgetPin (void) const; + bool GetUSBGadgetMode (void) const; // true if in USB gadget mode depending on USBGadget and USBGadgetPin + void SetUSBGadgetMode (bool USBGadgetMode); // Sound device const char *GetSoundDevice (void) const; @@ -192,6 +195,8 @@ public: private: CPropertiesFatFsFile m_Properties; + bool m_bUSBGadget; + unsigned m_nUSBGadgetPin; bool m_bUSBGadgetMode; std::string m_SoundDevice; diff --git a/src/kernel.cpp b/src/kernel.cpp index 33bc5f8..446747b 100644 --- a/src/kernel.cpp +++ b/src/kernel.cpp @@ -20,6 +20,7 @@ #include "kernel.h" #include #include +#include #include #include #include "usbminidexedmidigadget.h" @@ -92,8 +93,30 @@ bool CKernel::Initialize (void) m_pSPIMaster = nullptr; } } - - if (m_Config.GetUSBGadgetMode()) + + bool bUSBGadgetMode = false; + if (m_Config.GetUSBGadget()) + { + unsigned nUSBGadgetPin = m_Config.GetUSBGadgetPin(); + if (nUSBGadgetPin == 0) + { + // No hardware config option + bUSBGadgetMode = true; + } + else + { + // State of USB Gadget Mode determined by state of the pin. + // Pulled down = enable USB Gadget mode + CGPIOPin usbGadgetPin(nUSBGadgetPin, GPIOModeInputPullUp); + + if (usbGadgetPin.Read() == 0) + { + bUSBGadgetMode = true; + } + } + } + + if (bUSBGadgetMode) { #if RASPPI==5 #warning No support for USB Gadget Mode on RPI 5 yet @@ -107,6 +130,7 @@ bool CKernel::Initialize (void) // Run the USB stack in USB Host (default) mode m_pUSB = new CUSBHCIDevice (&mInterrupt, &mTimer, TRUE); } + m_Config.SetUSBGadgetMode(bUSBGadgetMode); if (!m_pUSB->Initialize ()) { diff --git a/src/minidexed.cpp b/src/minidexed.cpp index 429fddc..da9e598 100644 --- a/src/minidexed.cpp +++ b/src/minidexed.cpp @@ -102,17 +102,43 @@ CMiniDexed::CMiniDexed (CConfig *pConfig, CInterruptSystem *pInterrupt, m_pTG[i]->activate (); } - if (pConfig->GetUSBGadgetMode()) + unsigned nUSBGadgetPin = pConfig->GetUSBGadgetPin(); + bool bUSBGadget = pConfig->GetUSBGadget(); + bool bUSBGadgetMode = pConfig->GetUSBGadgetMode(); + + if (bUSBGadgetMode) { #if RASPPI==5 LOGNOTE ("USB Gadget (Device) Mode NOT supported on RPI 5"); #else - LOGNOTE ("USB In Gadget (Device) Mode"); + if (nUSBGadgetPin == 0) + { + LOGNOTE ("USB In Gadget (Device) Mode"); + } + else + { + LOGNOTE ("USB In Gadget (Device) Mode [USBGadgetPin %d = LOW]", nUSBGadgetPin); + } #endif } else { - LOGNOTE ("USB In Host Mode"); + if (bUSBGadget) + { + if (nUSBGadgetPin == 0) + { + // This shouldn't be possible... + LOGNOTE ("USB State Unknown"); + } + else + { + LOGNOTE ("USB In Host Mode [USBGadgetPin %d = HIGH]", nUSBGadgetPin); + } + } + else + { + LOGNOTE ("USB In Host Mode"); + } } for (unsigned i = 0; i < CConfig::MaxUSBMIDIDevices; i++) From dd5459eece66ca7292ae0b0aad032028775f8142 Mon Sep 17 00:00:00 2001 From: probonopd Date: Sat, 27 Jul 2024 11:53:20 +0200 Subject: [PATCH 17/38] Restrict permissions and pin "uses:" --- .github/workflows/pr-comment.yml | 8 +++++++- 1 file changed, 7 insertions(+), 1 deletion(-) diff --git a/.github/workflows/pr-comment.yml b/.github/workflows/pr-comment.yml index 74b0c0f..863a6d8 100644 --- a/.github/workflows/pr-comment.yml +++ b/.github/workflows/pr-comment.yml @@ -12,10 +12,16 @@ jobs: name: Add artifact links to PR and issues runs-on: ubuntu-22.04 + # Restrict permissions for the GITHUB_TOKEN, https://docs.github.com/en/actions/using-jobs/assigning-permissions-to-jobs + permissions: + issues: write + pull-requests: write + actions: read + steps: - name: Add artifact links to PR and issues if: github.event.workflow_run.event == 'pull_request' - uses: tonyhallett/artifacts-url-comments@v1.1.0 + uses: tonyhallett/artifacts-url-comments@0965ff1a7ae03c5c1644d3c30f956effea4e05ef # v1.1.0 env: GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} with: From f6ebd7871d0bc055224cf1b72278589a6be386ac Mon Sep 17 00:00:00 2001 From: Kevin <68612569+diyelectromusic@users.noreply.github.com> Date: Mon, 5 Aug 2024 08:39:32 +0100 Subject: [PATCH 18/38] Another attempt to support configuration TGs and polyphony (#690) * Initial commit for configuration TGs and polyphony across RPI1-5. * Ensure unused TGs in a performance are MIDI disabled. Set polyphony to higher defaults on Pi 4 and 5. * Actually, can just default to MIDI "disabled" directly in performance config if not present. * Fix issue with choosing max polyphony --- src/config.cpp | 56 ++++++++ src/config.h | 48 ++++++- src/mididevice.cpp | 10 +- src/mididevice.h | 2 +- src/minidexed.cpp | 284 +++++++++++++++++++++++++++----------- src/minidexed.h | 69 ++++----- src/performanceconfig.cpp | 126 +++++++++-------- src/performanceconfig.h | 62 +++++---- src/sysexfileloader.cpp | 17 +++ src/sysexfileloader.h | 1 + src/uimenu.cpp | 233 +++++++++++++++++++------------ src/uimenu.h | 6 +- src/userinterface.cpp | 2 +- 13 files changed, 618 insertions(+), 298 deletions(-) diff --git a/src/config.cpp b/src/config.cpp index 884d7cc..15a37ad 100644 --- a/src/config.cpp +++ b/src/config.cpp @@ -36,6 +36,20 @@ void CConfig::Load (void) { m_Properties.Load (); + // Number of Tone Generators and Polyphony + m_nToneGenerators = m_Properties.GetNumber ("ToneGenerators", DefToneGenerators); + m_nPolyphony = m_Properties.GetNumber ("Polyphony", DefaultNotes); + // At present there are only two options for tone generators: min or max + // and for the Pi 1,2,3 these are the same anyway. + if ((m_nToneGenerators != MinToneGenerators) && (m_nToneGenerators != AllToneGenerators)) + { + m_nToneGenerators = DefToneGenerators; + } + if (m_nPolyphony > MaxNotes) + { + m_nPolyphony = DefaultNotes; + } + m_bUSBGadget = m_Properties.GetNumber ("USBGadget", 0) != 0; m_nUSBGadgetPin = m_Properties.GetNumber ("USBGadgetPin", 0); // Default OFF SetUSBGadgetMode(m_bUSBGadget); // Might get overriden later by USBGadgetPin state @@ -177,6 +191,48 @@ void CConfig::Load (void) m_bPerformanceSelectChannel = m_Properties.GetNumber ("PerformanceSelectChannel", 0); } +unsigned CConfig::GetToneGenerators (void) const +{ + return m_nToneGenerators; +} + +unsigned CConfig::GetPolyphony (void) const +{ + return m_nPolyphony; +} + +unsigned CConfig::GetTGsCore1 (void) const +{ +#ifndef ARM_ALLOW_MULTI_CORE + return 0; +#else + if (m_nToneGenerators > MinToneGenerators) + { + return TGsCore1 + TGsCore1Opt; + } + else + { + return TGsCore1; + } +#endif +} + +unsigned CConfig::GetTGsCore23 (void) const +{ +#ifndef ARM_ALLOW_MULTI_CORE + return 0; +#else + if (m_nToneGenerators > MinToneGenerators) + { + return TGsCore23 + TGsCore23Opt; + } + else + { + return TGsCore23; + } +#endif +} + bool CConfig::GetUSBGadget (void) const { return m_bUSBGadget; diff --git a/src/config.h b/src/config.h index 7497c21..166c4bb 100644 --- a/src/config.h +++ b/src/config.h @@ -35,18 +35,49 @@ class CConfig // Configuration for MiniDexed { public: +// Set maximum, minimum and default numbers of tone generators, depending on Pi version. +// Actual number in can be changed via config settings for some Pis. #ifndef ARM_ALLOW_MULTI_CORE - static const unsigned ToneGenerators = 1; + // Pi V1 or Zero (single core) + static const unsigned MinToneGenerators = 1; + static const unsigned AllToneGenerators = 1; + static const unsigned DefToneGenerators = AllToneGenerators; #else +#if (RASPPI==4 || RASPPI==5) + // Pi 4 and 5 quad core + // These are max values, default is to support 8 in total with optional 16 TGs static const unsigned TGsCore1 = 2; // process 2 TGs on core 1 static const unsigned TGsCore23 = 3; // process 3 TGs on core 2 and 3 each - static const unsigned ToneGenerators = TGsCore1 + 2*TGsCore23; + static const unsigned TGsCore1Opt = 2; // process optional additional 2 TGs on core 1 + static const unsigned TGsCore23Opt = 3; // process optional additional 3 TGs on core 2 and 3 each + static const unsigned MinToneGenerators = TGsCore1 + 2*TGsCore23; + static const unsigned AllToneGenerators = TGsCore1 + TGsCore1Opt + 2*TGsCore23 + 2*TGsCore23Opt; + static const unsigned DefToneGenerators = MinToneGenerators; +#else + // Pi 2 or 3 quad core + static const unsigned TGsCore1 = 2; // process 2 TGs on core 1 + static const unsigned TGsCore23 = 3; // process 3 TGs on core 2 and 3 each + static const unsigned TGsCore1Opt = 0; + static const unsigned TGsCore23Opt = 0; + static const unsigned MinToneGenerators = TGsCore1 + 2*TGsCore23; + static const unsigned AllToneGenerators = MinToneGenerators; + static const unsigned DefToneGenerators = AllToneGenerators; #endif - +#endif + +// Set maximum polyphony, depending on PI version. This can be changed via config settings #if RASPPI == 1 - static const unsigned MaxNotes = 8; // polyphony + static const unsigned MaxNotes = 8; + static const unsigned DefaultNotes = 8; +#elif RASPPI == 4 + static const unsigned MaxNotes = 32; + static const unsigned DefaultNotes = 24; +#elif RASPPI == 5 + static const unsigned MaxNotes = 32; + static const unsigned DefaultNotes = 32; #else static const unsigned MaxNotes = 16; + static const unsigned DefaultNotes = 16; #endif static const unsigned MaxChunkSize = 4096; @@ -67,6 +98,12 @@ public: void Load (void); + // TGs and Polyphony + unsigned GetToneGenerators (void) const; + unsigned GetPolyphony (void) const; + unsigned GetTGsCore1 (void) const; + unsigned GetTGsCore23 (void) const; + // USB Mode bool GetUSBGadget (void) const; unsigned GetUSBGadgetPin (void) const; @@ -195,6 +232,9 @@ public: private: CPropertiesFatFsFile m_Properties; + unsigned m_nToneGenerators; + unsigned m_nPolyphony; + bool m_bUSBGadget; unsigned m_nUSBGadgetPin; bool m_bUSBGadgetMode; diff --git a/src/mididevice.cpp b/src/mididevice.cpp index 81314ba..504423e 100644 --- a/src/mididevice.cpp +++ b/src/mididevice.cpp @@ -65,7 +65,7 @@ CMIDIDevice::CMIDIDevice (CMiniDexed *pSynthesizer, CConfig *pConfig, CUserInter m_pConfig (pConfig), m_pUI (pUI) { - for (unsigned nTG = 0; nTG < CConfig::ToneGenerators; nTG++) + for (unsigned nTG = 0; nTG < CConfig::AllToneGenerators; nTG++) { m_ChannelMap[nTG] = Disabled; } @@ -78,13 +78,13 @@ CMIDIDevice::~CMIDIDevice (void) void CMIDIDevice::SetChannel (u8 ucChannel, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); m_ChannelMap[nTG] = ucChannel; } u8 CMIDIDevice::GetChannel (unsigned nTG) const { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); return m_ChannelMap[nTG]; } @@ -238,8 +238,8 @@ void CMIDIDevice::MIDIMessageHandler (const u8 *pMessage, size_t nLength, unsign break; } - // Process MIDI for each Tone Generator - for (unsigned nTG = 0; nTG < CConfig::ToneGenerators; nTG++) + // Process MIDI for each active Tone Generator + for (unsigned nTG = 0; nTG < m_pConfig->GetToneGenerators(); nTG++) { if (ucStatus == MIDI_SYSTEM_EXCLUSIVE_BEGIN) { diff --git a/src/mididevice.h b/src/mididevice.h index 0fc027c..b7e7fe7 100644 --- a/src/mididevice.h +++ b/src/mididevice.h @@ -65,7 +65,7 @@ private: CConfig *m_pConfig; CUserInterface *m_pUI; - u8 m_ChannelMap[CConfig::ToneGenerators]; + u8 m_ChannelMap[CConfig::AllToneGenerators]; std::string m_DeviceName; diff --git a/src/minidexed.cpp b/src/minidexed.cpp index da9e598..2e45f46 100644 --- a/src/minidexed.cpp +++ b/src/minidexed.cpp @@ -46,7 +46,7 @@ CMiniDexed::CMiniDexed (CConfig *pConfig, CInterruptSystem *pInterrupt, m_pSoundDevice (0), m_bChannelsSwapped (pConfig->GetChannelsSwapped ()), #ifdef ARM_ALLOW_MULTI_CORE - m_nActiveTGsLog2 (0), +// m_nActiveTGsLog2 (0), #endif m_GetChunkTimer ("GetChunk", 1000000U * pConfig->GetChunkSize ()/2 / pConfig->GetSampleRate ()), @@ -61,8 +61,12 @@ CMiniDexed::CMiniDexed (CConfig *pConfig, CInterruptSystem *pInterrupt, m_bLoadPerformanceBankBusy(false) { assert (m_pConfig); + + m_nToneGenerators = m_pConfig->GetToneGenerators(); + m_nPolyphony = m_pConfig->GetPolyphony(); + LOGNOTE("Tone Generators=%d, Polyphony=%d", m_nToneGenerators, m_nPolyphony); - for (unsigned i = 0; i < CConfig::ToneGenerators; i++) + for (unsigned i = 0; i < CConfig::AllToneGenerators; i++) { m_nVoiceBankID[i] = 0; m_nVoiceBankIDMSB[i] = 0; @@ -93,15 +97,20 @@ CMiniDexed::CMiniDexed (CConfig *pConfig, CInterruptSystem *pInterrupt, m_nAftertouchTarget[i]=0; m_nReverbSend[i] = 0; - m_uchOPMask[i] = 0b111111; // All operators on - m_pTG[i] = new CDexedAdapter (CConfig::MaxNotes, pConfig->GetSampleRate ()); - assert (m_pTG[i]); - - m_pTG[i]->setEngineType(pConfig->GetEngineType ()); - m_pTG[i]->activate (); + // Active the required number of active TGs + if (iGetSampleRate ()); + assert (m_pTG[i]); + + m_pTG[i]->setEngineType(pConfig->GetEngineType ()); + m_pTG[i]->activate (); + } } - + unsigned nUSBGadgetPin = pConfig->GetUSBGadgetPin(); bool bUSBGadget = pConfig->GetUSBGadget(); bool bUSBGadgetMode = pConfig->GetUSBGadgetMode(); @@ -156,6 +165,11 @@ CMiniDexed::CMiniDexed (CConfig *pConfig, CInterruptSystem *pInterrupt, // Quad DAC 8-channel mono only an option for RPI 5 m_bQuadDAC8Chan = pConfig->GetQuadDAC8Chan (); #endif + if (m_bQuadDAC8Chan && (m_nToneGenerators != 8)) + { + LOGNOTE("ERROR: Quad DAC Mode is only valid when number of TGs = 8. Defaulting to non-Quad DAC mode,"); + m_bQuadDAC8Chan = false; + } if (m_bQuadDAC8Chan) { LOGNOTE ("Configured for Quad DAC 8-channel Mono audio"); m_pSoundDevice = new CI2SSoundBaseDevice (pInterrupt, pConfig->GetSampleRate (), @@ -206,11 +220,11 @@ CMiniDexed::CMiniDexed (CConfig *pConfig, CInterruptSystem *pInterrupt, setMasterVolume(1.0); // BEGIN setup tg_mixer - tg_mixer = new AudioStereoMixer(pConfig->GetChunkSize()/2); + tg_mixer = new AudioStereoMixer(pConfig->GetChunkSize()/2); // END setup tgmixer // BEGIN setup reverb - reverb_send_mixer = new AudioStereoMixer(pConfig->GetChunkSize()/2); + reverb_send_mixer = new AudioStereoMixer(pConfig->GetChunkSize()/2); reverb = new AudioEffectPlateReverb(pConfig->GetSampleRate()); SetParameter (ParameterReverbEnable, 1); SetParameter (ParameterReverbSize, 70); @@ -261,7 +275,7 @@ bool CMiniDexed::Initialize (void) LOGNOTE("Program Change: Disabled"); } - for (unsigned i = 0; i < CConfig::ToneGenerators; i++) + for (unsigned i = 0; i < m_nToneGenerators; i++) { assert (m_pTG[i]); @@ -283,7 +297,7 @@ bool CMiniDexed::Initialize (void) reverb_send_mixer->gain(i,mapfloat(m_nReverbSend[i],0,99,0.0f,1.0f)); } - m_PerformanceConfig.Init(); + m_PerformanceConfig.Init(m_nToneGenerators); if (m_PerformanceConfig.Load ()) { LoadPerformanceParameters(); @@ -453,12 +467,16 @@ void CMiniDexed::Run (unsigned nCore) // process the TGs, assigned to this core (2 or 3) - assert (m_nFramesToProcess <= CConfig::MaxChunkSize); - unsigned nTG = CConfig::TGsCore1 + (nCore-2)*CConfig::TGsCore23; - for (unsigned i = 0; i < CConfig::TGsCore23; i++, nTG++) + assert (m_nFramesToProcess <= m_pConfig->MaxChunkSize); + unsigned nTG = m_pConfig->GetTGsCore1() + (nCore-2)*m_pConfig->GetTGsCore23(); + for (unsigned i = 0; i < m_pConfig->GetTGsCore23(); i++, nTG++) { - assert (m_pTG[nTG]); - m_pTG[nTG]->getSamples (m_OutputLevel[nTG],m_nFramesToProcess); + assert (nTG < CConfig::AllToneGenerators); + if (nTG < m_pConfig->GetToneGenerators()) + { + assert (m_pTG[nTG]); + m_pTG[nTG]->getSamples (m_OutputLevel[nTG],m_nFramesToProcess); + } } } } @@ -480,7 +498,8 @@ void CMiniDexed::BankSelect (unsigned nBank, unsigned nTG) { nBank=constrain((int)nBank,0,16383); - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG if (GetSysExFileLoader ()->IsValidBank(nBank)) { @@ -509,7 +528,9 @@ void CMiniDexed::BankSelectMSB (unsigned nBankMSB, unsigned nTG) { nBankMSB=constrain((int)nBankMSB,0,127); - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + // MIDI Spec 1.0 "BANK SELECT" states: // "The transmitter must transmit the MSB and LSB as a pair, // and the Program Change must be sent immediately after @@ -530,7 +551,9 @@ void CMiniDexed::BankSelectLSB (unsigned nBankLSB, unsigned nTG) { nBankLSB=constrain((int)nBankLSB,0,127); - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + unsigned nBank = m_nVoiceBankID[nTG]; unsigned nBankMSB = m_nVoiceBankIDMSB[nTG]; nBank = (nBankMSB << 7) + nBankLSB; @@ -578,7 +601,9 @@ void CMiniDexed::ProgramChange (unsigned nProgram, unsigned nTG) nProgram=constrain((int)nProgram,0,31); } - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + m_nProgram[nTG] = nProgram; uint8_t Buffer[156]; @@ -617,7 +642,9 @@ void CMiniDexed::SetVolume (unsigned nVolume, unsigned nTG) { nVolume=constrain((int)nVolume,0,127); - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + m_nVolume[nTG] = nVolume; assert (m_pTG[nTG]); @@ -630,7 +657,9 @@ void CMiniDexed::SetPan (unsigned nPan, unsigned nTG) { nPan=constrain((int)nPan,0,127); - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + m_nPan[nTG] = nPan; tg_mixer->pan(nTG,mapfloat(nPan,0,127,0.0f,1.0f)); @@ -643,7 +672,9 @@ void CMiniDexed::SetReverbSend (unsigned nReverbSend, unsigned nTG) { nReverbSend=constrain((int)nReverbSend,0,99); - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + m_nReverbSend[nTG] = nReverbSend; reverb_send_mixer->gain(nTG,mapfloat(nReverbSend,0,99,0.0f,1.0f)); @@ -655,7 +686,9 @@ void CMiniDexed::SetMasterTune (int nMasterTune, unsigned nTG) { nMasterTune=constrain((int)nMasterTune,-99,99); - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + m_nMasterTune[nTG] = nMasterTune; assert (m_pTG[nTG]); @@ -668,7 +701,9 @@ void CMiniDexed::SetCutoff (int nCutoff, unsigned nTG) { nCutoff = constrain (nCutoff, 0, 99); - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + m_nCutoff[nTG] = nCutoff; assert (m_pTG[nTG]); @@ -681,7 +716,9 @@ void CMiniDexed::SetResonance (int nResonance, unsigned nTG) { nResonance = constrain (nResonance, 0, 99); - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + m_nResonance[nTG] = nResonance; assert (m_pTG[nTG]); @@ -694,7 +731,9 @@ void CMiniDexed::SetResonance (int nResonance, unsigned nTG) void CMiniDexed::SetMIDIChannel (uint8_t uchChannel, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (uchChannel < CMIDIDevice::ChannelUnknown); m_nMIDIChannel[nTG] = uchChannel; @@ -713,6 +752,7 @@ void CMiniDexed::SetMIDIChannel (uint8_t uchChannel, unsigned nTG) } #ifdef ARM_ALLOW_MULTI_CORE +/* This doesn't appear to be used anywhere... unsigned nActiveTGs = 0; for (unsigned nTG = 0; nTG < CConfig::ToneGenerators; nTG++) { @@ -725,6 +765,7 @@ void CMiniDexed::SetMIDIChannel (uint8_t uchChannel, unsigned nTG) assert (nActiveTGs <= 8); static const unsigned Log2[] = {0, 0, 1, 2, 2, 3, 3, 3, 3}; m_nActiveTGsLog2 = Log2[nActiveTGs]; +*/ #endif m_UI.ParameterChanged (); @@ -732,7 +773,9 @@ void CMiniDexed::SetMIDIChannel (uint8_t uchChannel, unsigned nTG) void CMiniDexed::keyup (int16_t pitch, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); pitch = ApplyNoteLimits (pitch, nTG); @@ -744,7 +787,9 @@ void CMiniDexed::keyup (int16_t pitch, unsigned nTG) void CMiniDexed::keydown (int16_t pitch, uint8_t velocity, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); pitch = ApplyNoteLimits (pitch, nTG); @@ -756,7 +801,8 @@ void CMiniDexed::keydown (int16_t pitch, uint8_t velocity, unsigned nTG) int16_t CMiniDexed::ApplyNoteLimits (int16_t pitch, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return -1; // Not an active TG if ( pitch < (int16_t) m_nNoteLimitLow[nTG] || pitch > (int16_t) m_nNoteLimitHigh[nTG]) @@ -777,14 +823,18 @@ int16_t CMiniDexed::ApplyNoteLimits (int16_t pitch, unsigned nTG) void CMiniDexed::setSustain(bool sustain, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_pTG[nTG]->setSustain (sustain); } void CMiniDexed::panic(uint8_t value, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); if (value == 0) { m_pTG[nTG]->panic (); @@ -793,7 +843,9 @@ void CMiniDexed::panic(uint8_t value, unsigned nTG) void CMiniDexed::notesOff(uint8_t value, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); if (value == 0) { m_pTG[nTG]->notesOff (); @@ -802,7 +854,9 @@ void CMiniDexed::notesOff(uint8_t value, unsigned nTG) void CMiniDexed::setModWheel (uint8_t value, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_pTG[nTG]->setModWheel (value); } @@ -810,35 +864,45 @@ void CMiniDexed::setModWheel (uint8_t value, unsigned nTG) void CMiniDexed::setFootController (uint8_t value, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_pTG[nTG]->setFootController (value); } void CMiniDexed::setBreathController (uint8_t value, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_pTG[nTG]->setBreathController (value); } void CMiniDexed::setAftertouch (uint8_t value, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_pTG[nTG]->setAftertouch (value); } void CMiniDexed::setPitchbend (int16_t value, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_pTG[nTG]->setPitchbend (value); } void CMiniDexed::ControllersRefresh (unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_pTG[nTG]->ControllersRefresh (); } @@ -853,7 +917,7 @@ void CMiniDexed::SetParameter (TParameter Parameter, int nValue) switch (Parameter) { case ParameterCompressorEnable: - for (unsigned nTG = 0; nTG < CConfig::ToneGenerators; nTG++) + for (unsigned nTG = 0; nTG < m_nToneGenerators; nTG++) { assert (m_pTG[nTG]); m_pTG[nTG]->setCompressor (!!nValue); @@ -931,7 +995,8 @@ int CMiniDexed::GetParameter (TParameter Parameter) void CMiniDexed::SetTGParameter (TTGParameter Parameter, int nValue, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG switch (Parameter) { @@ -986,7 +1051,7 @@ void CMiniDexed::SetTGParameter (TTGParameter Parameter, int nValue, unsigned nT int CMiniDexed::GetTGParameter (TTGParameter Parameter, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); switch (Parameter) { @@ -1037,7 +1102,9 @@ int CMiniDexed::GetTGParameter (TTGParameter Parameter, unsigned nTG) void CMiniDexed::SetVoiceParameter (uint8_t uchOffset, uint8_t uchValue, unsigned nOP, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); assert (nOP <= 6); @@ -1070,7 +1137,9 @@ void CMiniDexed::SetVoiceParameter (uint8_t uchOffset, uint8_t uchValue, unsigne uint8_t CMiniDexed::GetVoiceParameter (uint8_t uchOffset, unsigned nOP, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return 0; // Not an active TG + assert (m_pTG[nTG]); assert (nOP <= 6); @@ -1094,13 +1163,15 @@ std::string CMiniDexed::GetVoiceName (unsigned nTG) { char VoiceName[11]; memset (VoiceName, 0, sizeof VoiceName); + VoiceName[0] = 32; // space + assert (nTG < CConfig::AllToneGenerators); - assert (nTG < CConfig::ToneGenerators); - assert (m_pTG[nTG]); - m_pTG[nTG]->setName (VoiceName); - + if (nTG < m_nToneGenerators) + { + assert (m_pTG[nTG]); + m_pTG[nTG]->setName (VoiceName); + } std::string Result (VoiceName); - return Result; } @@ -1142,6 +1213,7 @@ void CMiniDexed::ProcessSound (void) void CMiniDexed::ProcessSound (void) { assert (m_pSoundDevice); + assert (m_pConfig); unsigned nFrames = m_nQueueSizeFrames - m_pSoundDevice->GetQueueFramesAvail (); if (nFrames >= m_nQueueSizeFrames/2) @@ -1162,7 +1234,7 @@ void CMiniDexed::ProcessSound (void) // process the TGs assigned to core 1 assert (nFrames <= CConfig::MaxChunkSize); - for (unsigned i = 0; i < CConfig::TGsCore1; i++) + for (unsigned i = 0; i < m_pConfig->GetTGsCore1(); i++) { assert (m_pTG[i]); m_pTG[i]->getSamples (m_OutputLevel[i], nFrames); @@ -1181,9 +1253,10 @@ void CMiniDexed::ProcessSound (void) // Audio signal path after tone generators starts here // - assert (CConfig::ToneGenerators == 8); - if (m_bQuadDAC8Chan) { + // This is only supported when there are 8 TGs + assert (m_nToneGenerators == 8); + // No mixing is performed by MiniDexed, sound is output in 8 channels. // Note: one TG per audio channel; output=mono; no processing. const int Channels = 8; // One TG per channel @@ -1233,7 +1306,7 @@ void CMiniDexed::ProcessSound (void) if(nMasterVolume > 0.0) { - for (uint8_t i = 0; i < CConfig::ToneGenerators; i++) + for (uint8_t i = 0; i < m_nToneGenerators; i++) { tg_mixer->doAddMix(i,m_OutputLevel[i]); reverb_send_mixer->doAddMix(i,m_OutputLevel[i]); @@ -1358,7 +1431,7 @@ bool CMiniDexed::SavePerformance (bool bSaveAsDeault) bool CMiniDexed::DoSavePerformance (void) { - for (unsigned nTG = 0; nTG < CConfig::ToneGenerators; nTG++) + for (unsigned nTG = 0; nTG < CConfig::AllToneGenerators; nTG++) { m_PerformanceConfig.SetBankNumber (m_nVoiceBankID[nTG], nTG); m_PerformanceConfig.SetVoiceNumber (m_nProgram[nTG], nTG); @@ -1377,8 +1450,14 @@ bool CMiniDexed::DoSavePerformance (void) m_PerformanceConfig.SetNoteLimitLow (m_nNoteLimitLow[nTG], nTG); m_PerformanceConfig.SetNoteLimitHigh (m_nNoteLimitHigh[nTG], nTG); m_PerformanceConfig.SetNoteShift (m_nNoteShift[nTG], nTG); - m_pTG[nTG]->getVoiceData(m_nRawVoiceData); - m_PerformanceConfig.SetVoiceDataToTxt (m_nRawVoiceData, nTG); + if (nTG < m_pConfig->GetToneGenerators()) + { + m_pTG[nTG]->getVoiceData(m_nRawVoiceData); + } else { + // Not an active TG so provide default voice by asking for an invalid voice ID. + m_SysExFileLoader.GetVoice(CSysExFileLoader::MaxVoiceBankID, CSysExFileLoader::VoicesPerBank+1, m_nRawVoiceData); + } + m_PerformanceConfig.SetVoiceDataToTxt (m_nRawVoiceData, nTG); m_PerformanceConfig.SetMonoMode (m_bMonoMode[nTG], nTG); m_PerformanceConfig.SetModulationWheelRange (m_nModulationWheelRange[nTG], nTG); @@ -1412,7 +1491,9 @@ bool CMiniDexed::DoSavePerformance (void) void CMiniDexed::setMonoMode(uint8_t mono, uint8_t nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_bMonoMode[nTG]= mono != 0; m_pTG[nTG]->setMonoMode(constrain(mono, 0, 1)); @@ -1423,7 +1504,9 @@ void CMiniDexed::setMonoMode(uint8_t mono, uint8_t nTG) void CMiniDexed::setPitchbendRange(uint8_t range, uint8_t nTG) { range = constrain (range, 0, 12); - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_nPitchBendRange[nTG] = range; @@ -1435,7 +1518,9 @@ void CMiniDexed::setPitchbendRange(uint8_t range, uint8_t nTG) void CMiniDexed::setPitchbendStep(uint8_t step, uint8_t nTG) { step= constrain (step, 0, 12); - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_nPitchBendStep[nTG] = step; @@ -1448,7 +1533,9 @@ void CMiniDexed::setPortamentoMode(uint8_t mode, uint8_t nTG) { mode= constrain (mode, 0, 1); - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_nPortamentoMode[nTG] = mode; @@ -1460,7 +1547,9 @@ void CMiniDexed::setPortamentoMode(uint8_t mode, uint8_t nTG) void CMiniDexed::setPortamentoGlissando(uint8_t glissando, uint8_t nTG) { glissando = constrain (glissando, 0, 1); - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_nPortamentoGlissando[nTG] = glissando; @@ -1472,7 +1561,9 @@ void CMiniDexed::setPortamentoGlissando(uint8_t glissando, uint8_t nTG) void CMiniDexed::setPortamentoTime(uint8_t time, uint8_t nTG) { time = constrain (time, 0, 99); - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_nPortamentoTime[nTG] = time; @@ -1483,7 +1574,9 @@ void CMiniDexed::setPortamentoTime(uint8_t time, uint8_t nTG) void CMiniDexed::setModWheelRange(uint8_t range, uint8_t nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_nModulationWheelRange[nTG] = range; @@ -1496,7 +1589,9 @@ void CMiniDexed::setModWheelRange(uint8_t range, uint8_t nTG) void CMiniDexed::setModWheelTarget(uint8_t target, uint8_t nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_nModulationWheelTarget[nTG] = target; @@ -1508,7 +1603,9 @@ void CMiniDexed::setModWheelTarget(uint8_t target, uint8_t nTG) void CMiniDexed::setFootControllerRange(uint8_t range, uint8_t nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_nFootControlRange[nTG]=range; @@ -1521,7 +1618,9 @@ void CMiniDexed::setFootControllerRange(uint8_t range, uint8_t nTG) void CMiniDexed::setFootControllerTarget(uint8_t target, uint8_t nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_nFootControlTarget[nTG] = target; @@ -1533,7 +1632,9 @@ void CMiniDexed::setFootControllerTarget(uint8_t target, uint8_t nTG) void CMiniDexed::setBreathControllerRange(uint8_t range, uint8_t nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_nBreathControlRange[nTG]=range; @@ -1546,7 +1647,9 @@ void CMiniDexed::setBreathControllerRange(uint8_t range, uint8_t nTG) void CMiniDexed::setBreathControllerTarget(uint8_t target, uint8_t nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_nBreathControlTarget[nTG]=target; @@ -1558,7 +1661,9 @@ void CMiniDexed::setBreathControllerTarget(uint8_t target, uint8_t nTG) void CMiniDexed::setAftertouchRange(uint8_t range, uint8_t nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_nAftertouchRange[nTG]=range; @@ -1571,7 +1676,9 @@ void CMiniDexed::setAftertouchRange(uint8_t range, uint8_t nTG) void CMiniDexed::setAftertouchTarget(uint8_t target, uint8_t nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_nAftertouchTarget[nTG]=target; @@ -1583,7 +1690,9 @@ void CMiniDexed::setAftertouchTarget(uint8_t target, uint8_t nTG) void CMiniDexed::loadVoiceParameters(const uint8_t* data, uint8_t nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); uint8_t voice[161]; @@ -1604,7 +1713,9 @@ void CMiniDexed::loadVoiceParameters(const uint8_t* data, uint8_t nTG) void CMiniDexed::setVoiceDataElement(uint8_t data, uint8_t number, uint8_t nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); m_pTG[nTG]->setVoiceDataElement(constrain(data, 0, 155),constrain(number, 0, 99)); @@ -1614,7 +1725,9 @@ void CMiniDexed::setVoiceDataElement(uint8_t data, uint8_t number, uint8_t nTG) int16_t CMiniDexed::checkSystemExclusive(const uint8_t* pMessage,const uint16_t nLength, uint8_t nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return 0; // Not an active TG + assert (m_pTG[nTG]); return(m_pTG[nTG]->checkSystemExclusive(pMessage, nLength)); @@ -1625,10 +1738,17 @@ void CMiniDexed::getSysExVoiceDump(uint8_t* dest, uint8_t nTG) uint8_t checksum = 0; uint8_t data[155]; - assert (nTG < CConfig::ToneGenerators); - assert (m_pTG[nTG]); - - m_pTG[nTG]->getVoiceData(data); + assert (nTG < CConfig::AllToneGenerators); + if (nTG < m_nToneGenerators) + { + assert (m_pTG[nTG]); + m_pTG[nTG]->getVoiceData(data); + } + else + { + // Not an active TG so grab a default voice + m_SysExFileLoader.GetVoice(CSysExFileLoader::MaxVoiceBankID, CSysExFileLoader::VoicesPerBank+1, data); + } dest[0] = 0xF0; // SysEx start dest[1] = 0x43; // ID=Yamaha @@ -1791,7 +1911,7 @@ bool CMiniDexed::DoSavePerformanceNewFile (void) void CMiniDexed::LoadPerformanceParameters(void) { - for (unsigned nTG = 0; nTG < CConfig::ToneGenerators; nTG++) + for (unsigned nTG = 0; nTG < CConfig::AllToneGenerators; nTG++) { BankSelect (m_PerformanceConfig.GetBankNumber (nTG), nTG); @@ -1865,7 +1985,9 @@ bool CMiniDexed::IsValidPerformanceBank(unsigned nBankID) void CMiniDexed::SetVoiceName (std::string VoiceName, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); + if (nTG >= m_nToneGenerators) return; // Not an active TG + assert (m_pTG[nTG]); char Name[11]; strncpy(Name, VoiceName.c_str(),10); diff --git a/src/minidexed.h b/src/minidexed.h index 6e5e012..69dcf9c 100644 --- a/src/minidexed.h +++ b/src/minidexed.h @@ -230,7 +230,7 @@ public: private: int16_t ApplyNoteLimits (int16_t pitch, unsigned nTG); // returns < 0 to ignore note - uint8_t m_uchOPMask[CConfig::ToneGenerators]; + uint8_t m_uchOPMask[CConfig::AllToneGenerators]; void LoadPerformanceParameters(void); void ProcessSound (void); @@ -249,41 +249,44 @@ private: CConfig *m_pConfig; int m_nParameter[ParameterUnknown]; // global (non-TG) parameters + + unsigned m_nToneGenerators; + unsigned m_nPolyphony; - CDexedAdapter *m_pTG[CConfig::ToneGenerators]; + CDexedAdapter *m_pTG[CConfig::AllToneGenerators]; - unsigned m_nVoiceBankID[CConfig::ToneGenerators]; - unsigned m_nVoiceBankIDMSB[CConfig::ToneGenerators]; + unsigned m_nVoiceBankID[CConfig::AllToneGenerators]; + unsigned m_nVoiceBankIDMSB[CConfig::AllToneGenerators]; unsigned m_nVoiceBankIDPerformance; unsigned m_nVoiceBankIDMSBPerformance; - unsigned m_nProgram[CConfig::ToneGenerators]; - unsigned m_nVolume[CConfig::ToneGenerators]; - unsigned m_nPan[CConfig::ToneGenerators]; - int m_nMasterTune[CConfig::ToneGenerators]; - int m_nCutoff[CConfig::ToneGenerators]; - int m_nResonance[CConfig::ToneGenerators]; - unsigned m_nMIDIChannel[CConfig::ToneGenerators]; - unsigned m_nPitchBendRange[CConfig::ToneGenerators]; - unsigned m_nPitchBendStep[CConfig::ToneGenerators]; - unsigned m_nPortamentoMode[CConfig::ToneGenerators]; - unsigned m_nPortamentoGlissando[CConfig::ToneGenerators]; - unsigned m_nPortamentoTime[CConfig::ToneGenerators]; - bool m_bMonoMode[CConfig::ToneGenerators]; + unsigned m_nProgram[CConfig::AllToneGenerators]; + unsigned m_nVolume[CConfig::AllToneGenerators]; + unsigned m_nPan[CConfig::AllToneGenerators]; + int m_nMasterTune[CConfig::AllToneGenerators]; + int m_nCutoff[CConfig::AllToneGenerators]; + int m_nResonance[CConfig::AllToneGenerators]; + unsigned m_nMIDIChannel[CConfig::AllToneGenerators]; + unsigned m_nPitchBendRange[CConfig::AllToneGenerators]; + unsigned m_nPitchBendStep[CConfig::AllToneGenerators]; + unsigned m_nPortamentoMode[CConfig::AllToneGenerators]; + unsigned m_nPortamentoGlissando[CConfig::AllToneGenerators]; + unsigned m_nPortamentoTime[CConfig::AllToneGenerators]; + bool m_bMonoMode[CConfig::AllToneGenerators]; - unsigned m_nModulationWheelRange[CConfig::ToneGenerators]; - unsigned m_nModulationWheelTarget[CConfig::ToneGenerators]; - unsigned m_nFootControlRange[CConfig::ToneGenerators]; - unsigned m_nFootControlTarget[CConfig::ToneGenerators]; - unsigned m_nBreathControlRange[CConfig::ToneGenerators]; - unsigned m_nBreathControlTarget[CConfig::ToneGenerators]; - unsigned m_nAftertouchRange[CConfig::ToneGenerators]; - unsigned m_nAftertouchTarget[CConfig::ToneGenerators]; + unsigned m_nModulationWheelRange[CConfig::AllToneGenerators]; + unsigned m_nModulationWheelTarget[CConfig::AllToneGenerators]; + unsigned m_nFootControlRange[CConfig::AllToneGenerators]; + unsigned m_nFootControlTarget[CConfig::AllToneGenerators]; + unsigned m_nBreathControlRange[CConfig::AllToneGenerators]; + unsigned m_nBreathControlTarget[CConfig::AllToneGenerators]; + unsigned m_nAftertouchRange[CConfig::AllToneGenerators]; + unsigned m_nAftertouchTarget[CConfig::AllToneGenerators]; - unsigned m_nNoteLimitLow[CConfig::ToneGenerators]; - unsigned m_nNoteLimitHigh[CConfig::ToneGenerators]; - int m_nNoteShift[CConfig::ToneGenerators]; + unsigned m_nNoteLimitLow[CConfig::AllToneGenerators]; + unsigned m_nNoteLimitHigh[CConfig::AllToneGenerators]; + int m_nNoteShift[CConfig::AllToneGenerators]; - unsigned m_nReverbSend[CConfig::ToneGenerators]; + unsigned m_nReverbSend[CConfig::AllToneGenerators]; uint8_t m_nRawVoiceData[156]; @@ -305,18 +308,18 @@ private: unsigned m_nQueueSizeFrames; #ifdef ARM_ALLOW_MULTI_CORE - unsigned m_nActiveTGsLog2; +// unsigned m_nActiveTGsLog2; volatile TCoreStatus m_CoreStatus[CORES]; volatile unsigned m_nFramesToProcess; - float32_t m_OutputLevel[CConfig::ToneGenerators][CConfig::MaxChunkSize]; + float32_t m_OutputLevel[CConfig::AllToneGenerators][CConfig::MaxChunkSize]; #endif CPerformanceTimer m_GetChunkTimer; bool m_bProfileEnabled; AudioEffectPlateReverb* reverb; - AudioStereoMixer* tg_mixer; - AudioStereoMixer* reverb_send_mixer; + AudioStereoMixer* tg_mixer; + AudioStereoMixer* reverb_send_mixer; CSpinLock m_ReverbSpinLock; diff --git a/src/performanceconfig.cpp b/src/performanceconfig.cpp index 8cd6275..8a182e7 100644 --- a/src/performanceconfig.cpp +++ b/src/performanceconfig.cpp @@ -45,8 +45,20 @@ CPerformanceConfig::~CPerformanceConfig (void) { } -bool CPerformanceConfig::Init (void) +bool CPerformanceConfig::Init (unsigned nToneGenerators) { + // Different versions of Pi allow different TG configurations. + // On loading, performances will load up to the number of + // supported/active TGs. + // + // On saving, the active/supported number of TGs is used. + // + // This means that if an 8TG performance is loaded into + // a 16 TG system and then saved, the saved performance + // will include all 16 TG configurations. + // + m_nToneGenerators = nToneGenerators; + // Check intermal performance directory exists DIR Directory; FRESULT Result; @@ -95,7 +107,7 @@ bool CPerformanceConfig::Load (void) bool bResult = false; - for (unsigned nTG = 0; nTG < CConfig::ToneGenerators; nTG++) + for (unsigned nTG = 0; nTG < CConfig::AllToneGenerators; nTG++) { CString PropertyName; @@ -110,7 +122,7 @@ bool CPerformanceConfig::Load (void) } PropertyName.Format ("MIDIChannel%u", nTG+1); - unsigned nMIDIChannel = m_Properties.GetNumber (PropertyName, 255); + unsigned nMIDIChannel = m_Properties.GetNumber (PropertyName, 0); if (nMIDIChannel == 0) { m_nMIDIChannel[nTG] = CMIDIDevice::Disabled; @@ -217,7 +229,7 @@ bool CPerformanceConfig::Save (void) { m_Properties.RemoveAll (); - for (unsigned nTG = 0; nTG < CConfig::ToneGenerators; nTG++) + for (unsigned nTG = 0; nTG < m_nToneGenerators; nTG++) { CString PropertyName; @@ -333,145 +345,145 @@ bool CPerformanceConfig::Save (void) unsigned CPerformanceConfig::GetBankNumber (unsigned nTG) const { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); return m_nBankNumber[nTG]; } unsigned CPerformanceConfig::GetVoiceNumber (unsigned nTG) const { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); return m_nVoiceNumber[nTG]; } unsigned CPerformanceConfig::GetMIDIChannel (unsigned nTG) const { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); return m_nMIDIChannel[nTG]; } unsigned CPerformanceConfig::GetVolume (unsigned nTG) const { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); return m_nVolume[nTG]; } unsigned CPerformanceConfig::GetPan (unsigned nTG) const { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); return m_nPan[nTG]; } int CPerformanceConfig::GetDetune (unsigned nTG) const { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); return m_nDetune[nTG]; } unsigned CPerformanceConfig::GetCutoff (unsigned nTG) const { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); return m_nCutoff[nTG]; } unsigned CPerformanceConfig::GetResonance (unsigned nTG) const { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); return m_nResonance[nTG]; } unsigned CPerformanceConfig::GetNoteLimitLow (unsigned nTG) const { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); return m_nNoteLimitLow[nTG]; } unsigned CPerformanceConfig::GetNoteLimitHigh (unsigned nTG) const { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); return m_nNoteLimitHigh[nTG]; } int CPerformanceConfig::GetNoteShift (unsigned nTG) const { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); return m_nNoteShift[nTG]; } unsigned CPerformanceConfig::GetReverbSend (unsigned nTG) const { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); return m_nReverbSend[nTG]; } void CPerformanceConfig::SetBankNumber (unsigned nValue, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); m_nBankNumber[nTG] = nValue; } void CPerformanceConfig::SetVoiceNumber (unsigned nValue, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); m_nVoiceNumber[nTG] = nValue; } void CPerformanceConfig::SetMIDIChannel (unsigned nValue, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); m_nMIDIChannel[nTG] = nValue; } void CPerformanceConfig::SetVolume (unsigned nValue, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); m_nVolume[nTG] = nValue; } void CPerformanceConfig::SetPan (unsigned nValue, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); m_nPan[nTG] = nValue; } void CPerformanceConfig::SetDetune (int nValue, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); m_nDetune[nTG] = nValue; } void CPerformanceConfig::SetCutoff (unsigned nValue, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); m_nCutoff[nTG] = nValue; } void CPerformanceConfig::SetResonance (unsigned nValue, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); m_nResonance[nTG] = nValue; } void CPerformanceConfig::SetNoteLimitLow (unsigned nValue, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); m_nNoteLimitLow[nTG] = nValue; } void CPerformanceConfig::SetNoteLimitHigh (unsigned nValue, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); m_nNoteLimitHigh[nTG] = nValue; } void CPerformanceConfig::SetNoteShift (int nValue, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); m_nNoteShift[nTG] = nValue; } void CPerformanceConfig::SetReverbSend (unsigned nValue, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); m_nReverbSend[nTG] = nValue; } @@ -557,71 +569,71 @@ void CPerformanceConfig::SetReverbLevel (unsigned nValue) // Pitch bender and portamento: void CPerformanceConfig::SetPitchBendRange (unsigned nValue, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); m_nPitchBendRange[nTG] = nValue; } unsigned CPerformanceConfig::GetPitchBendRange (unsigned nTG) const { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); return m_nPitchBendRange[nTG]; } void CPerformanceConfig::SetPitchBendStep (unsigned nValue, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); m_nPitchBendStep[nTG] = nValue; } unsigned CPerformanceConfig::GetPitchBendStep (unsigned nTG) const { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); return m_nPitchBendStep[nTG]; } void CPerformanceConfig::SetPortamentoMode (unsigned nValue, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); m_nPortamentoMode[nTG] = nValue; } unsigned CPerformanceConfig::GetPortamentoMode (unsigned nTG) const { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); return m_nPortamentoMode[nTG]; } void CPerformanceConfig::SetPortamentoGlissando (unsigned nValue, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); m_nPortamentoGlissando[nTG] = nValue; } unsigned CPerformanceConfig::GetPortamentoGlissando (unsigned nTG) const { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); return m_nPortamentoGlissando[nTG]; } void CPerformanceConfig::SetPortamentoTime (unsigned nValue, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); m_nPortamentoTime[nTG] = nValue; } unsigned CPerformanceConfig::GetPortamentoTime (unsigned nTG) const { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); return m_nPortamentoTime[nTG]; } void CPerformanceConfig::SetMonoMode (bool bValue, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); m_bMonoMode[nTG] = bValue; } @@ -632,103 +644,103 @@ bool CPerformanceConfig::GetMonoMode (unsigned nTG) const void CPerformanceConfig::SetModulationWheelRange (unsigned nValue, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); m_nModulationWheelRange[nTG] = nValue; } unsigned CPerformanceConfig::GetModulationWheelRange (unsigned nTG) const { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); return m_nModulationWheelRange[nTG]; } void CPerformanceConfig::SetModulationWheelTarget (unsigned nValue, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); m_nModulationWheelTarget[nTG] = nValue; } unsigned CPerformanceConfig::GetModulationWheelTarget (unsigned nTG) const { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); return m_nModulationWheelTarget[nTG]; } void CPerformanceConfig::SetFootControlRange (unsigned nValue, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); m_nFootControlRange[nTG] = nValue; } unsigned CPerformanceConfig::GetFootControlRange (unsigned nTG) const { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); return m_nFootControlRange[nTG]; } void CPerformanceConfig::SetFootControlTarget (unsigned nValue, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); m_nFootControlTarget[nTG] = nValue; } unsigned CPerformanceConfig::GetFootControlTarget (unsigned nTG) const { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); return m_nFootControlTarget[nTG]; } void CPerformanceConfig::SetBreathControlRange (unsigned nValue, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); m_nBreathControlRange[nTG] = nValue; } unsigned CPerformanceConfig::GetBreathControlRange (unsigned nTG) const { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); return m_nBreathControlRange[nTG]; } void CPerformanceConfig::SetBreathControlTarget (unsigned nValue, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); m_nBreathControlTarget[nTG] = nValue; } unsigned CPerformanceConfig::GetBreathControlTarget (unsigned nTG) const { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); return m_nBreathControlTarget[nTG]; } void CPerformanceConfig::SetAftertouchRange (unsigned nValue, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); m_nAftertouchRange[nTG] = nValue; } unsigned CPerformanceConfig::GetAftertouchRange (unsigned nTG) const { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); return m_nAftertouchRange[nTG]; } void CPerformanceConfig::SetAftertouchTarget (unsigned nValue, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); m_nAftertouchTarget[nTG] = nValue; } unsigned CPerformanceConfig::GetAftertouchTarget (unsigned nTG) const { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); return m_nAftertouchTarget[nTG]; } void CPerformanceConfig::SetVoiceDataToTxt (const uint8_t *pData, unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); m_nVoiceDataTxt[nTG] = ""; char nDtoH[]="0123456789ABCDEF"; for (int i = 0; i < NUM_VOICE_PARAM; i++) @@ -744,7 +756,7 @@ void CPerformanceConfig::SetVoiceDataToTxt (const uint8_t *pData, unsigned nTG) uint8_t *CPerformanceConfig::GetVoiceDataFromTxt (unsigned nTG) { - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); static uint8_t pData[NUM_VOICE_PARAM]; std::string nHtoD="0123456789ABCDEF"; diff --git a/src/performanceconfig.h b/src/performanceconfig.h index 0d50daa..c9c4b27 100644 --- a/src/performanceconfig.h +++ b/src/performanceconfig.h @@ -36,7 +36,7 @@ public: CPerformanceConfig (FATFS *pFileSystem); ~CPerformanceConfig (void); - bool Init (void); + bool Init (unsigned nToneGenerators); bool Load (void); @@ -151,35 +151,37 @@ public: private: CPropertiesFatFsFile m_Properties; - - unsigned m_nBankNumber[CConfig::ToneGenerators]; - unsigned m_nVoiceNumber[CConfig::ToneGenerators]; - unsigned m_nMIDIChannel[CConfig::ToneGenerators]; - unsigned m_nVolume[CConfig::ToneGenerators]; - unsigned m_nPan[CConfig::ToneGenerators]; - int m_nDetune[CConfig::ToneGenerators]; - unsigned m_nCutoff[CConfig::ToneGenerators]; - unsigned m_nResonance[CConfig::ToneGenerators]; - unsigned m_nNoteLimitLow[CConfig::ToneGenerators]; - unsigned m_nNoteLimitHigh[CConfig::ToneGenerators]; - int m_nNoteShift[CConfig::ToneGenerators]; - int m_nReverbSend[CConfig::ToneGenerators]; - unsigned m_nPitchBendRange[CConfig::ToneGenerators]; - unsigned m_nPitchBendStep[CConfig::ToneGenerators]; - unsigned m_nPortamentoMode[CConfig::ToneGenerators]; - unsigned m_nPortamentoGlissando[CConfig::ToneGenerators]; - unsigned m_nPortamentoTime[CConfig::ToneGenerators]; - std::string m_nVoiceDataTxt[CConfig::ToneGenerators]; - bool m_bMonoMode[CConfig::ToneGenerators]; - - unsigned m_nModulationWheelRange[CConfig::ToneGenerators]; - unsigned m_nModulationWheelTarget[CConfig::ToneGenerators]; - unsigned m_nFootControlRange[CConfig::ToneGenerators]; - unsigned m_nFootControlTarget[CConfig::ToneGenerators]; - unsigned m_nBreathControlRange[CConfig::ToneGenerators]; - unsigned m_nBreathControlTarget[CConfig::ToneGenerators]; - unsigned m_nAftertouchRange[CConfig::ToneGenerators]; - unsigned m_nAftertouchTarget[CConfig::ToneGenerators]; + + unsigned m_nToneGenerators; + + unsigned m_nBankNumber[CConfig::AllToneGenerators]; + unsigned m_nVoiceNumber[CConfig::AllToneGenerators]; + unsigned m_nMIDIChannel[CConfig::AllToneGenerators]; + unsigned m_nVolume[CConfig::AllToneGenerators]; + unsigned m_nPan[CConfig::AllToneGenerators]; + int m_nDetune[CConfig::AllToneGenerators]; + unsigned m_nCutoff[CConfig::AllToneGenerators]; + unsigned m_nResonance[CConfig::AllToneGenerators]; + unsigned m_nNoteLimitLow[CConfig::AllToneGenerators]; + unsigned m_nNoteLimitHigh[CConfig::AllToneGenerators]; + int m_nNoteShift[CConfig::AllToneGenerators]; + int m_nReverbSend[CConfig::AllToneGenerators]; + unsigned m_nPitchBendRange[CConfig::AllToneGenerators]; + unsigned m_nPitchBendStep[CConfig::AllToneGenerators]; + unsigned m_nPortamentoMode[CConfig::AllToneGenerators]; + unsigned m_nPortamentoGlissando[CConfig::AllToneGenerators]; + unsigned m_nPortamentoTime[CConfig::AllToneGenerators]; + std::string m_nVoiceDataTxt[CConfig::AllToneGenerators]; + bool m_bMonoMode[CConfig::AllToneGenerators]; + + unsigned m_nModulationWheelRange[CConfig::AllToneGenerators]; + unsigned m_nModulationWheelTarget[CConfig::AllToneGenerators]; + unsigned m_nFootControlRange[CConfig::AllToneGenerators]; + unsigned m_nFootControlTarget[CConfig::AllToneGenerators]; + unsigned m_nBreathControlRange[CConfig::AllToneGenerators]; + unsigned m_nBreathControlTarget[CConfig::AllToneGenerators]; + unsigned m_nAftertouchRange[CConfig::AllToneGenerators]; + unsigned m_nAftertouchTarget[CConfig::AllToneGenerators]; unsigned m_nLastPerformance; unsigned m_nActualPerformance = 0; diff --git a/src/sysexfileloader.cpp b/src/sysexfileloader.cpp index ea5ba5c..6b83f5b 100644 --- a/src/sysexfileloader.cpp +++ b/src/sysexfileloader.cpp @@ -273,6 +273,23 @@ std::string CSysExFileLoader::GetBankName (unsigned nBankID) return "NO NAME"; } +std::string CSysExFileLoader::GetVoiceName (unsigned nBankID, unsigned nVoiceID) +{ + if ((nBankID <= MaxVoiceBankID) && (nVoiceID < VoicesPerBank)) + { + if (IsValidBank(nBankID)) + { + // The name is the last 10 characters of the voice data + char sVoiceName[11]; + strncpy (sVoiceName, (char *)((char *)&(m_pVoiceBank[nBankID]->Voice[nVoiceID]) + SizePackedVoice - 10), 10); + sVoiceName[10] = 0; + std::string result(sVoiceName); + return result; + } + } + return "INIT VOICE"; +} + unsigned CSysExFileLoader::GetNextBankUp (unsigned nBankID) { // Find the next loaded bank "up" from the provided bank ID diff --git a/src/sysexfileloader.h b/src/sysexfileloader.h index 4918db6..272d775 100644 --- a/src/sysexfileloader.h +++ b/src/sysexfileloader.h @@ -60,6 +60,7 @@ public: void Load (bool bHeaderlessSysExVoices = false); std::string GetBankName (unsigned nBankID); // 0 .. MaxVoiceBankID + std::string GetVoiceName (unsigned nBankID, unsigned nVoice); // 0 .. MaxVoiceBankID, 0 .. VoicesPerBank-1 unsigned GetNumHighestBank (); // 0 .. MaxVoiceBankID bool IsValidBank (unsigned nBankID); unsigned GetNextBankUp (unsigned nBankID); diff --git a/src/uimenu.cpp b/src/uimenu.cpp index 82a426a..8b72425 100644 --- a/src/uimenu.cpp +++ b/src/uimenu.cpp @@ -51,6 +51,16 @@ const CUIMenu::TMenuItem CUIMenu::s_MainMenu[] = {"TG6", MenuHandler, s_TGMenu, 5}, {"TG7", MenuHandler, s_TGMenu, 6}, {"TG8", MenuHandler, s_TGMenu, 7}, +#if (RASPPI==4 || RASPPI==5) + {"TG9", MenuHandler, s_TGMenu, 8}, + {"TG10", MenuHandler, s_TGMenu, 9}, + {"TG11", MenuHandler, s_TGMenu, 10}, + {"TG12", MenuHandler, s_TGMenu, 11}, + {"TG13", MenuHandler, s_TGMenu, 12}, + {"TG14", MenuHandler, s_TGMenu, 13}, + {"TG15", MenuHandler, s_TGMenu, 14}, + {"TG16", MenuHandler, s_TGMenu, 15}, +#endif #endif {"Effects", MenuHandler, s_EffectsMenu}, {"Performance", MenuHandler, s_PerformanceMenu}, @@ -334,9 +344,10 @@ const CUIMenu::TMenuItem CUIMenu::s_PerformanceMenu[] = }; -CUIMenu::CUIMenu (CUserInterface *pUI, CMiniDexed *pMiniDexed) +CUIMenu::CUIMenu (CUserInterface *pUI, CMiniDexed *pMiniDexed, CConfig *pConfig) : m_pUI (pUI), m_pMiniDexed (pMiniDexed), + m_pConfig (pConfig), m_pParentMenu (s_MenuRoot), m_pCurrentMenu (s_MainMenu), m_nCurrentMenuItem (0), @@ -344,23 +355,27 @@ CUIMenu::CUIMenu (CUserInterface *pUI, CMiniDexed *pMiniDexed) m_nCurrentParameter (0), m_nCurrentMenuDepth (0) { -#ifndef ARM_ALLOW_MULTI_CORE - // If there is just one core, then there is only a single - // tone generator so start on the TG1 menu... - m_pParentMenu = s_MainMenu; - m_pCurrentMenu = s_TGMenu; - m_nCurrentMenuItem = 0; - m_nCurrentSelection = 0; - m_nCurrentParameter = 0; - m_nCurrentMenuDepth = 1; + assert (m_pConfig); + m_nToneGenerators = m_pConfig->GetToneGenerators(); - // Place the "root" menu at the top of the stack - m_MenuStackParent[0] = s_MenuRoot; - m_MenuStackMenu[0] = s_MainMenu; - m_nMenuStackItem[0] = 0; - m_nMenuStackSelection[0] = 0; - m_nMenuStackParameter[0] = 0; -#endif + if (m_nToneGenerators == 1) + { + // If there is just one core, then there is only a single + // tone generator so start on the TG1 menu... + m_pParentMenu = s_MainMenu; + m_pCurrentMenu = s_TGMenu; + m_nCurrentMenuItem = 0; + m_nCurrentSelection = 0; + m_nCurrentParameter = 0; + m_nCurrentMenuDepth = 1; + + // Place the "root" menu at the top of the stack + m_MenuStackParent[0] = s_MenuRoot; + m_MenuStackMenu[0] = s_MainMenu; + m_nMenuStackItem[0] = 0; + m_nMenuStackSelection[0] = 0; + m_nMenuStackParameter[0] = 0; + } } void CUIMenu::EventHandler (TMenuEvent Event) @@ -383,28 +398,31 @@ void CUIMenu::EventHandler (TMenuEvent Event) break; case MenuEventHome: -#ifdef ARM_ALLOW_MULTI_CORE - m_pParentMenu = s_MenuRoot; - m_pCurrentMenu = s_MainMenu; - m_nCurrentMenuItem = 0; - m_nCurrentSelection = 0; - m_nCurrentParameter = 0; - m_nCurrentMenuDepth = 0; -#else - // "Home" is the TG0 menu if only one TG active - m_pParentMenu = s_MainMenu; - m_pCurrentMenu = s_TGMenu; - m_nCurrentMenuItem = 0; - m_nCurrentSelection = 0; - m_nCurrentParameter = 0; - m_nCurrentMenuDepth = 1; - // Place the "root" menu at the top of the stack - m_MenuStackParent[0] = s_MenuRoot; - m_MenuStackMenu[0] = s_MainMenu; - m_nMenuStackItem[0] = 0; - m_nMenuStackSelection[0] = 0; - m_nMenuStackParameter[0] = 0; -#endif + if (m_nToneGenerators == 1) + { + // "Home" is the TG0 menu if only one TG active + m_pParentMenu = s_MainMenu; + m_pCurrentMenu = s_TGMenu; + m_nCurrentMenuItem = 0; + m_nCurrentSelection = 0; + m_nCurrentParameter = 0; + m_nCurrentMenuDepth = 1; + // Place the "root" menu at the top of the stack + m_MenuStackParent[0] = s_MenuRoot; + m_MenuStackMenu[0] = s_MainMenu; + m_nMenuStackItem[0] = 0; + m_nMenuStackSelection[0] = 0; + m_nMenuStackParameter[0] = 0; + } + else + { + m_pParentMenu = s_MenuRoot; + m_pCurrentMenu = s_MainMenu; + m_nCurrentMenuItem = 0; + m_nCurrentSelection = 0; + m_nCurrentParameter = 0; + m_nCurrentMenuDepth = 0; + } EventHandler (MenuEventUpdate); break; @@ -453,7 +471,30 @@ void CUIMenu::MenuHandler (CUIMenu *pUIMenu, TMenuEvent Event) break; case MenuEventStepDown: - if (pUIMenu->m_nCurrentSelection > 0) + if (pUIMenu->m_nCurrentSelection == 0) + { + // If in main mennu, wrap around + if (pUIMenu->m_pCurrentMenu == s_MainMenu) + { + // Find last entry with a name + while (pUIMenu->m_pCurrentMenu[pUIMenu->m_nCurrentSelection+1].Name) + { + pUIMenu->m_nCurrentSelection++; + } + } + } + else if (pUIMenu->m_nCurrentSelection > 0) + { + pUIMenu->m_nCurrentSelection--; + } + // Might need to trim menu if number of TGs is configured to be less than the maximum supported + while ((pUIMenu->m_pCurrentMenu == s_MainMenu) && (pUIMenu->m_nCurrentSelection > 0) && + ( // Skip any unused menus + (pUIMenu->m_pCurrentMenu[pUIMenu->m_nCurrentSelection].MenuItem == s_TGMenu) && + (pUIMenu->m_pCurrentMenu[pUIMenu->m_nCurrentSelection].Parameter >= pUIMenu->m_nToneGenerators) && + (pUIMenu->m_pCurrentMenu[pUIMenu->m_nCurrentSelection].Parameter < CConfig::AllToneGenerators) + ) + ) { pUIMenu->m_nCurrentSelection--; } @@ -463,7 +504,27 @@ void CUIMenu::MenuHandler (CUIMenu *pUIMenu, TMenuEvent Event) ++pUIMenu->m_nCurrentSelection; if (!pUIMenu->m_pCurrentMenu[pUIMenu->m_nCurrentSelection].Name) // more entries? { - pUIMenu->m_nCurrentSelection--; + if (pUIMenu->m_pCurrentMenu == s_MainMenu) + { + // If in main mennu, wrap around + pUIMenu->m_nCurrentSelection = 0; + } + else + { + // Return to last known good item + pUIMenu->m_nCurrentSelection--; + } + } + // Might need to trim menu if number of TGs is configured to be less than the maximum supported + while ((pUIMenu->m_pCurrentMenu == s_MainMenu) && (pUIMenu->m_pCurrentMenu[pUIMenu->m_nCurrentSelection+1].Name) && + ( // Skip any unused TG menus + (pUIMenu->m_pCurrentMenu[pUIMenu->m_nCurrentSelection].MenuItem == s_TGMenu) && + (pUIMenu->m_pCurrentMenu[pUIMenu->m_nCurrentSelection].Parameter >= pUIMenu->m_nToneGenerators) && + (pUIMenu->m_pCurrentMenu[pUIMenu->m_nCurrentSelection].Parameter < CConfig::AllToneGenerators) + ) + ) + { + pUIMenu->m_nCurrentSelection++; } break; @@ -1151,7 +1212,7 @@ void CUIMenu::TGShortcutHandler (TMenuEvent Event) assert (m_nCurrentMenuDepth >= 2); assert (m_MenuStackMenu[0] = s_MainMenu); unsigned nTG = m_nMenuStackSelection[0]; - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); assert (m_nMenuStackItem[1] == nTG); assert (m_nMenuStackParameter[1] == nTG); @@ -1166,7 +1227,7 @@ void CUIMenu::TGShortcutHandler (TMenuEvent Event) nTG++; } - if (nTG < CConfig::ToneGenerators) + if (nTG < m_nToneGenerators) { m_nMenuStackSelection[0] = nTG; m_nMenuStackItem[1] = nTG; @@ -1262,51 +1323,53 @@ void CUIMenu::PgmUpDownHandler (TMenuEvent Event) if (m_MenuStackMenu[0] == s_MainMenu && (m_pCurrentMenu == s_TGMenu) || (m_MenuStackMenu[1] == s_TGMenu)) { nTG = m_nMenuStackSelection[0]; } - assert (nTG < CConfig::ToneGenerators); - - int nPgm = m_pMiniDexed->GetTGParameter (CMiniDexed::TGParameterProgram, nTG); - - assert (Event == MenuEventPgmDown || Event == MenuEventPgmUp); - if (Event == MenuEventPgmDown) + assert (nTG < CConfig::AllToneGenerators); + if (nTG < m_nToneGenerators) { - //LOGNOTE("PgmDown"); - if (--nPgm < 0) + int nPgm = m_pMiniDexed->GetTGParameter (CMiniDexed::TGParameterProgram, nTG); + + assert (Event == MenuEventPgmDown || Event == MenuEventPgmUp); + if (Event == MenuEventPgmDown) { - // Switch down a voice bank and set to the last voice - nPgm = CSysExFileLoader::VoicesPerBank-1; - int nVB = m_pMiniDexed->GetTGParameter(CMiniDexed::TGParameterVoiceBank, nTG); - nVB = m_pMiniDexed->GetSysExFileLoader ()->GetNextBankDown(nVB); - m_pMiniDexed->SetTGParameter (CMiniDexed::TGParameterVoiceBank, nVB, nTG); + //LOGNOTE("PgmDown"); + if (--nPgm < 0) + { + // Switch down a voice bank and set to the last voice + nPgm = CSysExFileLoader::VoicesPerBank-1; + int nVB = m_pMiniDexed->GetTGParameter(CMiniDexed::TGParameterVoiceBank, nTG); + nVB = m_pMiniDexed->GetSysExFileLoader ()->GetNextBankDown(nVB); + m_pMiniDexed->SetTGParameter (CMiniDexed::TGParameterVoiceBank, nVB, nTG); + } + m_pMiniDexed->SetTGParameter (CMiniDexed::TGParameterProgram, nPgm, nTG); } - m_pMiniDexed->SetTGParameter (CMiniDexed::TGParameterProgram, nPgm, nTG); - } - else - { - //LOGNOTE("PgmUp"); - if (++nPgm > (int) CSysExFileLoader::VoicesPerBank-1) + else { - // Switch up a voice bank and reset to voice 0 - nPgm = 0; - int nVB = m_pMiniDexed->GetTGParameter(CMiniDexed::TGParameterVoiceBank, nTG); - nVB = m_pMiniDexed->GetSysExFileLoader ()->GetNextBankUp(nVB); - m_pMiniDexed->SetTGParameter (CMiniDexed::TGParameterVoiceBank, nVB, nTG); + //LOGNOTE("PgmUp"); + if (++nPgm > (int) CSysExFileLoader::VoicesPerBank-1) + { + // Switch up a voice bank and reset to voice 0 + nPgm = 0; + int nVB = m_pMiniDexed->GetTGParameter(CMiniDexed::TGParameterVoiceBank, nTG); + nVB = m_pMiniDexed->GetSysExFileLoader ()->GetNextBankUp(nVB); + m_pMiniDexed->SetTGParameter (CMiniDexed::TGParameterVoiceBank, nVB, nTG); + } + m_pMiniDexed->SetTGParameter (CMiniDexed::TGParameterProgram, nPgm, nTG); } - m_pMiniDexed->SetTGParameter (CMiniDexed::TGParameterProgram, nPgm, nTG); - } - // Skip empty voices. - // Use same criteria in EditProgramNumber () too. - string voiceName = m_pMiniDexed->GetVoiceName (nTG); - if (voiceName == "EMPTY " - || voiceName == " " - || voiceName == "----------" - || voiceName == "~~~~~~~~~~" ) - { - if (Event == MenuEventPgmUp) { - PgmUpDownHandler (MenuEventPgmUp); - } - if (Event == MenuEventPgmDown) { - PgmUpDownHandler (MenuEventPgmDown); + // Skip empty voices. + // Use same criteria in EditProgramNumber () too. + string voiceName = m_pMiniDexed->GetVoiceName (nTG); + if (voiceName == "EMPTY " + || voiceName == " " + || voiceName == "----------" + || voiceName == "~~~~~~~~~~" ) + { + if (Event == MenuEventPgmUp) { + PgmUpDownHandler (MenuEventPgmUp); + } + if (Event == MenuEventPgmDown) { + PgmUpDownHandler (MenuEventPgmDown); + } } } } @@ -1317,7 +1380,7 @@ void CUIMenu::TGUpDownHandler (TMenuEvent Event) // This will update the menus to position it for the next TG up or down unsigned nTG = 0; - if (CConfig::ToneGenerators <= 1) { + if (m_nToneGenerators <= 1) { // Nothing to do if only a single TG return; } @@ -1328,7 +1391,7 @@ void CUIMenu::TGUpDownHandler (TMenuEvent Event) nTG = m_nMenuStackSelection[0]; } - assert (nTG < CConfig::ToneGenerators); + assert (nTG < CConfig::AllToneGenerators); assert (Event == MenuEventTGDown || Event == MenuEventTGUp); if (Event == MenuEventTGDown) { @@ -1340,7 +1403,7 @@ void CUIMenu::TGUpDownHandler (TMenuEvent Event) else { //LOGNOTE("TGUp"); - if (nTG < CConfig::ToneGenerators - 1) { + if (nTG < m_nToneGenerators - 1) { nTG++; } } diff --git a/src/uimenu.h b/src/uimenu.h index d5b48dc..0034278 100644 --- a/src/uimenu.h +++ b/src/uimenu.h @@ -25,6 +25,7 @@ #include #include +#include "config.h" class CMiniDexed; class CUserInterface; @@ -53,7 +54,7 @@ public: }; public: - CUIMenu (CUserInterface *pUI, CMiniDexed *pMiniDexed); + CUIMenu (CUserInterface *pUI, CMiniDexed *pMiniDexed, CConfig *pConfig); void EventHandler (TMenuEvent Event); @@ -128,6 +129,9 @@ private: private: CUserInterface *m_pUI; CMiniDexed *m_pMiniDexed; + CConfig *m_pConfig; + + unsigned m_nToneGenerators; const TMenuItem *m_pParentMenu; const TMenuItem *m_pCurrentMenu; diff --git a/src/userinterface.cpp b/src/userinterface.cpp index 9e7c112..aa46f9e 100644 --- a/src/userinterface.cpp +++ b/src/userinterface.cpp @@ -38,7 +38,7 @@ CUserInterface::CUserInterface (CMiniDexed *pMiniDexed, CGPIOManager *pGPIOManag m_pUIButtons (0), m_pRotaryEncoder (0), m_bSwitchPressed (false), - m_Menu (this, pMiniDexed) + m_Menu (this, pMiniDexed, pConfig) { } From e03122a3d232e7b20e5ccbb54de6f0dff5f7666f Mon Sep 17 00:00:00 2001 From: probonopd Date: Wed, 7 Aug 2024 18:24:32 +0200 Subject: [PATCH 19/38] Stargazers over time [ci skip] --- README.md | 3 +++ 1 file changed, 3 insertions(+) diff --git a/README.md b/README.md index dcb9d84..68cdb02 100644 --- a/README.md +++ b/README.md @@ -98,3 +98,6 @@ This project stands on the shoulders of giants. Special thanks to: - [Banana71](https://github.com/Banana71) for the sound design of the [Soundplantage](https://github.com/Banana71/Soundplantage) performances shipped with MiniDexed - [BobanSpasic](https://github.com/BobanSpasic) for the [MiniDexedLibrarian](https://github.com/BobanSpasic/MiniDexedLibrarian) software, [MiniDexed performance converter](https://github.com/BobanSpasic/MDX_PerfConv) and [collection of performances for MiniDexed](https://github.com/BobanSpasic/MDX_Vault) - [diyelectromusic](https://github.com/diyelectromusic/) for many [contributions](https://github.com/probonopd/MiniDexed/commits?author=diyelectromusic) + +## Stargazers over time +[![Stargazers over time](https://starchart.cc/probonopd/MiniDexed.svg?variant=adaptive)](https://starchart.cc/probonopd/MiniDexed) From 6c56960888c1ab144f0229ca3d0c323027302a3f Mon Sep 17 00:00:00 2001 From: probonopd Date: Sun, 8 Sep 2024 13:58:41 +0200 Subject: [PATCH 20/38] Lower debounce time (#714) https://github.com/probonopd/MiniDexed/issues/713 --- src/uibuttons.h | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/uibuttons.h b/src/uibuttons.h index 47d128d..44dcca0 100644 --- a/src/uibuttons.h +++ b/src/uibuttons.h @@ -26,7 +26,7 @@ #include "config.h" #define BUTTONS_UPDATE_NUM_TICKS 100 -#define DEBOUNCE_TIME 100 +#define DEBOUNCE_TIME 20 #define MAX_GPIO_BUTTONS 9 // 5 UI buttons, 4 Program/TG Select buttons #define MAX_MIDI_BUTTONS 9 #define MAX_BUTTONS (MAX_GPIO_BUTTONS+MAX_MIDI_BUTTONS) From 8c18e60e5133a56e7208c4ff051cc2ec80e0aaa1 Mon Sep 17 00:00:00 2001 From: Kevin <68612569+diyelectromusic@users.noreply.github.com> Date: Sun, 29 Sep 2024 10:29:26 +0100 Subject: [PATCH 21/38] Initial implementation of system-level, i.e. all TG, MIDI control maps (#704) * Initial implementation of system-level, i.e. all TG, MIDI control maps * Ensure msg only handled once by the first TG with a matching receiving MIDI channel. * Optimise system CC handling esp when unrecognised CCs are received. * Update in optimisation of System CC handling. --- src/config.cpp | 23 +++++++++- src/config.h | 10 ++++- src/mididevice.cpp | 104 ++++++++++++++++++++++++++++++++++++++++++++- src/mididevice.h | 9 ++++ 4 files changed, 141 insertions(+), 5 deletions(-) diff --git a/src/config.cpp b/src/config.cpp index 15a37ad..5214713 100644 --- a/src/config.cpp +++ b/src/config.cpp @@ -108,6 +108,10 @@ void CConfig::Load (void) m_bMIDIAutoVoiceDumpOnPC = m_Properties.GetNumber ("MIDIAutoVoiceDumpOnPC", 0) != 0; m_bHeaderlessSysExVoices = m_Properties.GetNumber ("HeaderlessSysExVoices", 0) != 0; m_bExpandPCAcrossBanks = m_Properties.GetNumber ("ExpandPCAcrossBanks", 1) != 0; + + m_nMIDISystemCCVol = m_Properties.GetNumber ("MIDISystemCCVol", 0); + m_nMIDISystemCCPan = m_Properties.GetNumber ("MIDISystemCCPan", 0); + m_nMIDISystemCCDetune = m_Properties.GetNumber ("MIDISystemCCDetune", 0); m_bLCDEnabled = m_Properties.GetNumber ("LCDEnabled", 0) != 0; m_nLCDPinEnable = m_Properties.GetNumber ("LCDPinEnable", 4); @@ -283,6 +287,11 @@ unsigned CConfig::GetEngineType (void) const return m_EngineType; } +bool CConfig::GetQuadDAC8Chan (void) const +{ + return m_bQuadDAC8Chan; +} + unsigned CConfig::GetMIDIBaudRate (void) const { return m_nMIDIBaudRate; @@ -323,9 +332,19 @@ bool CConfig::GetExpandPCAcrossBanks (void) const return m_bExpandPCAcrossBanks; } -bool CConfig::GetQuadDAC8Chan (void) const +unsigned CConfig::GetMIDISystemCCVol (void) const { - return m_bQuadDAC8Chan; + return m_nMIDISystemCCVol; +} + +unsigned CConfig::GetMIDISystemCCPan (void) const +{ + return m_nMIDISystemCCPan; +} + +unsigned CConfig::GetMIDISystemCCDetune (void) const +{ + return m_nMIDISystemCCDetune; } bool CConfig::GetLCDEnabled (void) const diff --git a/src/config.h b/src/config.h index 166c4bb..a454d90 100644 --- a/src/config.h +++ b/src/config.h @@ -117,6 +117,7 @@ public: unsigned GetDACI2CAddress (void) const; // 0 for auto probing bool GetChannelsSwapped (void) const; unsigned GetEngineType (void) const; + bool GetQuadDAC8Chan (void) const; // false if not specified // MIDI unsigned GetMIDIBaudRate (void) const; @@ -127,7 +128,9 @@ public: bool GetMIDIAutoVoiceDumpOnPC (void) const; // false if not specified bool GetHeaderlessSysExVoices (void) const; // false if not specified bool GetExpandPCAcrossBanks (void) const; // true if not specified - bool GetQuadDAC8Chan (void) const; // false if not specified + unsigned GetMIDISystemCCVol (void) const; + unsigned GetMIDISystemCCPan (void) const; + unsigned GetMIDISystemCCDetune (void) const; // HD44780 LCD // GPIO pin numbers are chip numbers, not header positions @@ -245,6 +248,7 @@ private: unsigned m_nDACI2CAddress; bool m_bChannelsSwapped; unsigned m_EngineType; + bool m_bQuadDAC8Chan; unsigned m_nMIDIBaudRate; std::string m_MIDIThruIn; @@ -254,7 +258,9 @@ private: bool m_bMIDIAutoVoiceDumpOnPC; bool m_bHeaderlessSysExVoices; bool m_bExpandPCAcrossBanks; - bool m_bQuadDAC8Chan; + unsigned m_nMIDISystemCCVol; + unsigned m_nMIDISystemCCPan; + unsigned m_nMIDISystemCCDetune; bool m_bLCDEnabled; unsigned m_nLCDPinEnable; diff --git a/src/mididevice.cpp b/src/mididevice.cpp index 504423e..f2b51de 100644 --- a/src/mididevice.cpp +++ b/src/mididevice.cpp @@ -53,6 +53,21 @@ LOGMODULE ("mididevice"); #define MIDI_PROGRAM_CHANGE 0b1100 #define MIDI_PITCH_BEND 0b1110 +// MIDI "System" level (i.e. all TG) custom CC maps +// Note: Even if number of TGs is not 8, there are only 8 +// available to be used in the mappings here. +#define NUM_MIDI_CC_MAPS 8 +const unsigned MIDISystemCCMap[NUM_MIDI_CC_MAPS][8] = { + {0,0,0,0,0,0,0,0}, // 0 = disabled + {16,17,18,19,80,81,82,83}, // 1 = General Purpose Controllers 1-8 + {20,21,22,23,24,25,26,27}, + {52,53,54,55,56,57,58,59}, + {102,103,104,105,106,107,108,109}, + {110,111,112,113,114,115,116,117}, + {3,9,14,15,28,29,30,31}, + {35,41,46,47,60,61,62,63} +}; + #define MIDI_SYSTEM_EXCLUSIVE_BEGIN 0xF0 #define MIDI_SYSTEM_EXCLUSIVE_END 0xF7 #define MIDI_TIMING_CLOCK 0xF8 @@ -69,6 +84,34 @@ CMIDIDevice::CMIDIDevice (CMiniDexed *pSynthesizer, CConfig *pConfig, CUserInter { m_ChannelMap[nTG] = Disabled; } + + m_nMIDISystemCCVol = m_pConfig->GetMIDISystemCCVol(); + m_nMIDISystemCCPan = m_pConfig->GetMIDISystemCCPan(); + m_nMIDISystemCCDetune = m_pConfig->GetMIDISystemCCDetune(); + + m_MIDISystemCCBitmap[0] = 0; + m_MIDISystemCCBitmap[1] = 0; + m_MIDISystemCCBitmap[2] = 0; + m_MIDISystemCCBitmap[3] = 0; + + for (int tg=0; tg<8; tg++) + { + if (m_nMIDISystemCCVol != 0) { + u8 cc = MIDISystemCCMap[m_nMIDISystemCCVol][tg]; + m_MIDISystemCCBitmap[cc>>5] |= (1<<(cc%32)); + } + if (m_nMIDISystemCCPan != 0) { + u8 cc = MIDISystemCCMap[m_nMIDISystemCCPan][tg]; + m_MIDISystemCCBitmap[cc>>5] |= (1<<(cc%32)); + } + if (m_nMIDISystemCCDetune != 0) { + u8 cc = MIDISystemCCMap[m_nMIDISystemCCDetune][tg]; + m_MIDISystemCCBitmap[cc>>5] |= (1<<(cc%32)); + } + } + if (m_pConfig->GetMIDIDumpEnabled ()) { + LOGNOTE("MIDI System CC Map: %08X %08X %08X %08X", m_MIDISystemCCBitmap[3],m_MIDISystemCCBitmap[2],m_MIDISystemCCBitmap[1],m_MIDISystemCCBitmap[0]); + } } CMIDIDevice::~CMIDIDevice (void) @@ -239,7 +282,9 @@ void CMIDIDevice::MIDIMessageHandler (const u8 *pMessage, size_t nLength, unsign } // Process MIDI for each active Tone Generator - for (unsigned nTG = 0; nTG < m_pConfig->GetToneGenerators(); nTG++) + bool bSystemCCHandled = false; + bool bSystemCCChecked = false; + for (unsigned nTG = 0; nTG < m_pConfig->GetToneGenerators() && !bSystemCCHandled; nTG++) { if (ucStatus == MIDI_SYSTEM_EXCLUSIVE_BEGIN) { @@ -373,6 +418,17 @@ void CMIDIDevice::MIDIMessageHandler (const u8 *pMessage, size_t nLength, unsign m_pSynthesizer->notesOff (pMessage[2], nTG); } break; + + default: + // Check for system-level, cross-TG MIDI Controls, but only do it once. + // Also, if successfully handled, then no need to process other TGs, + // so it is possible to break out of the main TG loop too. + // Note: We handle this here so we get the TG MIDI channel checking. + if (!bSystemCCChecked) { + bSystemCCHandled = HandleMIDISystemCC(pMessage[1], pMessage[2]); + bSystemCCChecked = true; + } + break; } break; @@ -418,6 +474,52 @@ void CMIDIDevice::AddDevice (const char *pDeviceName) s_DeviceMap.insert (std::pair (pDeviceName, this)); } +bool CMIDIDevice::HandleMIDISystemCC(const u8 ucCC, const u8 ucCCval) +{ + // This only makes sense when there are at least 8 TGs. + // Note: If more than 8 TGs then only 8 TGs are controllable this way. + if (m_pConfig->GetToneGenerators() < 8) { + return false; + } + + // Quickly reject any CCs not in the configured maps + if ((m_MIDISystemCCBitmap[ucCC>>5] & (1<<(ucCC%32))) == 0) { + // Not in the map + return false; + } + + // Not looking for duplicate CCs so return once handled + for (unsigned tg=0; tg<8; tg++) { + if (m_nMIDISystemCCVol != 0) { + if (ucCC == MIDISystemCCMap[m_nMIDISystemCCVol][tg]) { + m_pSynthesizer->SetVolume (ucCCval, tg); + return true; + } + } + if (m_nMIDISystemCCPan != 0) { + if (ucCC == MIDISystemCCMap[m_nMIDISystemCCPan][tg]) { + m_pSynthesizer->SetPan (ucCCval, tg); + return true; + } + } + if (m_nMIDISystemCCDetune != 0) { + if (ucCC == MIDISystemCCMap[m_nMIDISystemCCDetune][tg]) { + if (ucCCval == 0) + { + m_pSynthesizer->SetMasterTune (0, tg); + } + else + { + m_pSynthesizer->SetMasterTune (maplong (ucCCval, 1, 127, -99, 99), tg); + } + return true; + } + } + } + + return false; +} + void CMIDIDevice::HandleSystemExclusive(const uint8_t* pMessage, const size_t nLength, const unsigned nCable, const uint8_t nTG) { int16_t sysex_return; diff --git a/src/mididevice.h b/src/mididevice.h index b7e7fe7..44f1691 100644 --- a/src/mididevice.h +++ b/src/mididevice.h @@ -60,12 +60,21 @@ protected: void MIDIMessageHandler (const u8 *pMessage, size_t nLength, unsigned nCable = 0); void AddDevice (const char *pDeviceName); void HandleSystemExclusive(const uint8_t* pMessage, const size_t nLength, const unsigned nCable, const uint8_t nTG); + +private: + bool HandleMIDISystemCC(const u8 ucCC, const u8 ucCCval); + private: CMiniDexed *m_pSynthesizer; CConfig *m_pConfig; CUserInterface *m_pUI; u8 m_ChannelMap[CConfig::AllToneGenerators]; + + unsigned m_nMIDISystemCCVol; + unsigned m_nMIDISystemCCPan; + unsigned m_nMIDISystemCCDetune; + u32 m_MIDISystemCCBitmap[4]; // to allow for 128 bit entries std::string m_DeviceName; From 3a1f7e5f00703127f8a6372578274b8a5a2ea809 Mon Sep 17 00:00:00 2001 From: probonopd Date: Sat, 9 Nov 2024 09:35:34 +0100 Subject: [PATCH 22/38] Add Network section to minidexed.ini https://github.com/probonopd/MiniDexed/pull/744#issuecomment-2466114284 --- src/minidexed.ini | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/src/minidexed.ini b/src/minidexed.ini index de9f150..41c5a12 100644 --- a/src/minidexed.ini +++ b/src/minidexed.ini @@ -137,5 +137,10 @@ EncoderPinData=9 MIDIDumpEnabled=0 ProfileEnabled=0 +# Network +NetworkEnabled=0 +NetworkDHCP=1 +NetworkType=wifi + # Performance PerformanceSelectToLoad=1 From bb1f352fdb37de252012d28dc8b6d8b3034947d2 Mon Sep 17 00:00:00 2001 From: probonopd Date: Sat, 9 Nov 2024 08:52:20 +0000 Subject: [PATCH 23/38] Add WLAN firmware and wpa_supplicant.conf --- .github/workflows/build.yml | 8 +++++++- Synth_Dexed | 2 +- circle-stdlib | 2 +- 3 files changed, 9 insertions(+), 3 deletions(-) diff --git a/.github/workflows/build.yml b/.github/workflows/build.yml index 8df5911..ff616dd 100644 --- a/.github/workflows/build.yml +++ b/.github/workflows/build.yml @@ -57,7 +57,7 @@ jobs: export PATH=$(readlink -f ./gcc-*arm-none*/bin/):$PATH RPI=1 bash -ex build.sh cp ./src/kernel*.img ./kernels/ - - name: Get Raspberry Pi boot files + - name: Get Raspberry Pi boot files and WLAN firmware run: | set -ex export PATH=$(readlink -f ./gcc-*aarch64-none*/bin/):$PATH @@ -74,6 +74,12 @@ jobs: cd sdcard cp ../kernels/* . || true cd - + # WLAN firmware + cp circle-stdlib/libs/circle/addon/wlan/sample/hello_wlan/wpa_supplicant.conf sdcard/ + mkdir -p sdcard/firmware + cd sdcard/firmware + make -f ../../circle-stdlib/libs/circle/addon/wlan/firmware/Makefile + cd - - name: Get performance files run: | git clone https://github.com/Banana71/Soundplantage --depth 1 # depth 1 means only the latest commit diff --git a/Synth_Dexed b/Synth_Dexed index 8c677ce..c9f5274 160000 --- a/Synth_Dexed +++ b/Synth_Dexed @@ -1 +1 @@ -Subproject commit 8c677ceb4b3fb73f8643e30ff6cf4158dc8b9e53 +Subproject commit c9f52741a802ad9bb01263823650f7cc3b0b5108 diff --git a/circle-stdlib b/circle-stdlib index 61cf3a4..3bd135d 160000 --- a/circle-stdlib +++ b/circle-stdlib @@ -1 +1 @@ -Subproject commit 61cf3a47bf93628039078b7c840e44432e52343e +Subproject commit 3bd135d35ac23c7c726924cb85b5890cbf80bbfe From f5447152d40bd49f173f474ea33de0a4d41bd43a Mon Sep 17 00:00:00 2001 From: probonopd Date: Sat, 9 Nov 2024 09:05:01 +0000 Subject: [PATCH 24/38] Use "WLAN" terminology --- src/minidexed.cpp | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/minidexed.cpp b/src/minidexed.cpp index ceb3491..a1a12d5 100644 --- a/src/minidexed.cpp +++ b/src/minidexed.cpp @@ -2236,7 +2236,7 @@ bool CMiniDexed::InitNetwork() if (m_pConfig->GetNetworkEnabled () && (strcmp(m_pConfig->GetNetworkType(), "wifi") == 0)) { - LOGNOTE("Initializing Wi-Fi"); + LOGNOTE("Initializing WLAN"); if (m_WLAN.Initialize() && m_WPASupplicant.Initialize()) { @@ -2245,7 +2245,7 @@ bool CMiniDexed::InitNetwork() } else - LOGERR("Failed to initialize Wi-Fi"); + LOGERR("Failed to initialize WLAN"); } else if (m_pConfig->GetNetworkEnabled () && (strcmp(m_pConfig->GetNetworkType(), "ethernet") == 0)) { From 0b69e35d4db0f76fab049afd8d623137cd454350 Mon Sep 17 00:00:00 2001 From: probonopd Date: Sat, 9 Nov 2024 11:39:54 +0100 Subject: [PATCH 25/38] Update submod.sh --- submod.sh | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/submod.sh b/submod.sh index f9524a3..559aeb0 100755 --- a/submod.sh +++ b/submod.sh @@ -6,13 +6,13 @@ git submodule update --init --recursive # # Use fixed master branch of circle-stdlib then re-update cd circle-stdlib/ -git checkout 3bd135d +git checkout 695ab4a git submodule update --init --recursive cd - # # Optional update submodules explicitly cd circle-stdlib/libs/circle -git checkout fff3764 +git checkout tags/Step48 cd - cd circle-stdlib/libs/circle-newlib #git checkout develop From 084dca96f6bc1a5bcf89d9a3e2c05acf4d19b6c3 Mon Sep 17 00:00:00 2001 From: probonopd Date: Sat, 9 Nov 2024 11:40:08 +0100 Subject: [PATCH 26/38] Add initial mdns responder for rtp-midi aka applemidi (#749) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Co-authored-by: Ömer Şiar Baysal --- src/minidexed.cpp | 19 +++++++++++++------ src/minidexed.h | 1 + src/rtpmididevice.cpp | 14 ++++---------- src/rtpmididevice.h | 22 +--------------------- 4 files changed, 19 insertions(+), 37 deletions(-) diff --git a/src/minidexed.cpp b/src/minidexed.cpp index a1a12d5..c210536 100644 --- a/src/minidexed.cpp +++ b/src/minidexed.cpp @@ -2197,7 +2197,8 @@ void CMiniDexed::UpdateNetwork() CString IPString; m_pNet->GetConfig()->GetIPAddress()->Format(&IPString); - LOGNOTE("Network up and running at: %s", static_cast(IPString)); + //LOGNOTE("Network up and running at: %s", static_cast(IPString)); + m_UDPMIDI.Initialize(); @@ -2214,11 +2215,17 @@ void CMiniDexed::UpdateNetwork() LOGNOTE("FTP daemon initialized"); } - m_UI.DisplayWrite ("IP", - "Network", - IPString, - 0, - 1); + m_UI.DisplayWrite ("IP", "Network", IPString, 0, 1); + + CmDNSPublisher *pmDNSPublisher = new CmDNSPublisher (m_pNet); + assert (pmDNSPublisher); + static const char ServiceName[] = "minidexed-rtpmidi"; + static const char *ppText[] = {"RTP-MIDI Receiver", nullptr}; // TXT record strings + if (!pmDNSPublisher->PublishService (ServiceName, CmDNSPublisher::ServiceTypeAppleMIDI, + 5004, ppText)) + { + LOGPANIC ("Cannot publish mdns service"); + } } else if (m_bNetworkReady && !bNetIsRunning) { diff --git a/src/minidexed.h b/src/minidexed.h index 2d73e74..8466eb0 100644 --- a/src/minidexed.h +++ b/src/minidexed.h @@ -43,6 +43,7 @@ #include #include #include +#include #include #include "common.h" #include "effect_mixer.hpp" diff --git a/src/rtpmididevice.cpp b/src/rtpmididevice.cpp index 2c29a94..572a5c9 100644 --- a/src/rtpmididevice.cpp +++ b/src/rtpmididevice.cpp @@ -26,24 +26,21 @@ #include "rtpmididevice.h" #include +#define VIRTUALCABLE 24 + LOGMODULE("rtpmididevice"); CRTPMIDIDevice::CRTPMIDIDevice (CMiniDexed *pSynthesizer, CConfig *pConfig, CUserInterface *pUI) : CMIDIDevice (pSynthesizer, pConfig, pUI), m_pConfig (pConfig) - - //m_Serial (pInterrupt, TRUE), - //m_nSerialState (0), - //m_nSysEx (0), - //m_SendBuffer (&m_Serial) { AddDevice ("rtpdummy"); } CRTPMIDIDevice::~CRTPMIDIDevice (void) { - //m_nSerialState = 255; + } boolean CRTPMIDIDevice::Initialize (void) @@ -63,10 +60,7 @@ boolean CRTPMIDIDevice::Initialize (void) void CRTPMIDIDevice::OnAppleMIDIDataReceived(const u8* pData, size_t nSize) { - LOGNOTE("Recieved RTP MIDI Data"); - printf ("MIDI-RTP: %02X %02X\n", - (unsigned) pData[0], (unsigned) pData[1]); - MIDIMessageHandler(pData, nSize); + MIDIMessageHandler(pData, nSize, VIRTUALCABLE); } void CRTPMIDIDevice::OnAppleMIDIConnect(const CIPAddress* pIPAddress, const char* pName) diff --git a/src/rtpmididevice.h b/src/rtpmididevice.h index 418fcde..870301b 100644 --- a/src/rtpmididevice.h +++ b/src/rtpmididevice.h @@ -27,11 +27,6 @@ #include "config.h" #include "net/applemidi.h" -#include -#include -#include -#include - class CMiniDexed; class CRTPMIDIDevice : CAppleMIDIHandler, CMIDIDevice @@ -44,26 +39,11 @@ public: virtual void OnAppleMIDIDataReceived(const u8* pData, size_t nSize) override; virtual void OnAppleMIDIConnect(const CIPAddress* pIPAddress, const char* pName) override; virtual void OnAppleMIDIDisconnect(const CIPAddress* pIPAddress, const char* pName) override; - //void OnAppleMIDIDataReceived(const u8* pData, size_t nSize); - //void OnAppleMIDIConnect(const CIPAddress* pIPAddress, const char* pName); - //void OnAppleMIDIDisconnect(const CIPAddress* pIPAddress, const char* pName); - - //void Process (void); - - //void Send (const u8 *pMessage, size_t nLength, unsigned nCable = 0) override; private: CConfig *m_pConfig; - - //CSerialDevice m_Serial; - //unsigned m_nSerialState; - //unsigned m_nSysEx; - //u8 m_SerialMessage[MAX_MIDI_MESSAGE]; - - //CWriteBufferDevice m_SendBuffer; CBcmRandomNumberGenerator m_Random; - //CAppleMIDIHandler* m_MIDIHandler; - CAppleMIDIParticipant* m_pAppleMIDIParticipant; // AppleMIDI participant instance + CAppleMIDIParticipant* m_pAppleMIDIParticipant; }; #endif From 3a673dd48cb37974424d4cde0e01f59388b6b9bf Mon Sep 17 00:00:00 2001 From: probonopd Date: Sat, 9 Nov 2024 11:44:58 +0100 Subject: [PATCH 27/38] circle: git checkout c243194 --- submod.sh | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/submod.sh b/submod.sh index 559aeb0..b18a43d 100755 --- a/submod.sh +++ b/submod.sh @@ -12,7 +12,7 @@ cd - # # Optional update submodules explicitly cd circle-stdlib/libs/circle -git checkout tags/Step48 +git checkout c243194 cd - cd circle-stdlib/libs/circle-newlib #git checkout develop From d9e32a117a28a10ca03bbab0269a849f4fba6109 Mon Sep 17 00:00:00 2001 From: probonopd Date: Sat, 9 Nov 2024 11:49:49 +0100 Subject: [PATCH 28/38] circle: git checkout 59484ba (develop) --- submod.sh | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/submod.sh b/submod.sh index b18a43d..02a9e36 100755 --- a/submod.sh +++ b/submod.sh @@ -12,7 +12,7 @@ cd - # # Optional update submodules explicitly cd circle-stdlib/libs/circle -git checkout c243194 +git checkout 59484ba cd - cd circle-stdlib/libs/circle-newlib #git checkout develop From c5e338e1fc68597ef784dcd7610f92f847b1b62c Mon Sep 17 00:00:00 2001 From: probonopd Date: Sat, 9 Nov 2024 11:58:45 +0100 Subject: [PATCH 29/38] Update submod.sh --- submod.sh | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/submod.sh b/submod.sh index 02a9e36..0f7a869 100755 --- a/submod.sh +++ b/submod.sh @@ -6,13 +6,13 @@ git submodule update --init --recursive # # Use fixed master branch of circle-stdlib then re-update cd circle-stdlib/ -git checkout 695ab4a +git checkout 3bd135d git submodule update --init --recursive cd - # # Optional update submodules explicitly cd circle-stdlib/libs/circle -git checkout 59484ba +git checkout 3bd135d cd - cd circle-stdlib/libs/circle-newlib #git checkout develop From 8d26e20441f65e43d95d8acf8b80ae7f872c408a Mon Sep 17 00:00:00 2001 From: probonopd Date: Sat, 9 Nov 2024 11:59:59 +0100 Subject: [PATCH 30/38] circle: git checkout c243194 (master) --- submod.sh | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/submod.sh b/submod.sh index 0f7a869..78531dc 100755 --- a/submod.sh +++ b/submod.sh @@ -12,7 +12,7 @@ cd - # # Optional update submodules explicitly cd circle-stdlib/libs/circle -git checkout 3bd135d +git checkout c243194 cd - cd circle-stdlib/libs/circle-newlib #git checkout develop From 2a6d966aa16047420508c35afd863ae67c4ae52e Mon Sep 17 00:00:00 2001 From: probonopd Date: Sat, 9 Nov 2024 12:38:01 +0100 Subject: [PATCH 31/38] Use "MiniDexed" as the mDNS-SD service name --- src/minidexed.cpp | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/minidexed.cpp b/src/minidexed.cpp index c210536..5059ea0 100644 --- a/src/minidexed.cpp +++ b/src/minidexed.cpp @@ -2219,7 +2219,7 @@ void CMiniDexed::UpdateNetwork() CmDNSPublisher *pmDNSPublisher = new CmDNSPublisher (m_pNet); assert (pmDNSPublisher); - static const char ServiceName[] = "minidexed-rtpmidi"; + static const char ServiceName[] = "MiniDexed"; static const char *ppText[] = {"RTP-MIDI Receiver", nullptr}; // TXT record strings if (!pmDNSPublisher->PublishService (ServiceName, CmDNSPublisher::ServiceTypeAppleMIDI, 5004, ppText)) @@ -2284,4 +2284,4 @@ bool CMiniDexed::InitNetwork() m_pNetDevice = CNetDevice::GetNetDevice(NetDeviceType); } return m_pNet != nullptr; -} \ No newline at end of file +} From a2f8b4c5cfeb8dd54bb9c1ffb153b7789d409acc Mon Sep 17 00:00:00 2001 From: probonopd Date: Sat, 9 Nov 2024 12:45:20 +0100 Subject: [PATCH 32/38] Try to also publish _ftp._tcp --- src/minidexed.cpp | 7 +++++++ 1 file changed, 7 insertions(+) diff --git a/src/minidexed.cpp b/src/minidexed.cpp index 5059ea0..2b4127d 100644 --- a/src/minidexed.cpp +++ b/src/minidexed.cpp @@ -2226,6 +2226,13 @@ void CMiniDexed::UpdateNetwork() { LOGPANIC ("Cannot publish mdns service"); } + static const char *ppText2[] = {"FTP Server", nullptr}; // TXT record strings + static constexpr const char *ServiceTypeFTP = "_ftp._tcp"; + if (!pmDNSPublisher->PublishService (ServiceName, CmDNSPublisher::ServiceTypeFTP, + 21, ppText2)) + { + LOGPANIC ("Cannot publish mdns service"); + } } else if (m_bNetworkReady && !bNetIsRunning) { From 20a6ec8cd3f117fbd84cf144eff2db1f0d564946 Mon Sep 17 00:00:00 2001 From: probonopd Date: Sat, 9 Nov 2024 12:51:58 +0100 Subject: [PATCH 33/38] ServiceTypeFTP --- src/minidexed.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/minidexed.cpp b/src/minidexed.cpp index 2b4127d..ddde684 100644 --- a/src/minidexed.cpp +++ b/src/minidexed.cpp @@ -2228,7 +2228,7 @@ void CMiniDexed::UpdateNetwork() } static const char *ppText2[] = {"FTP Server", nullptr}; // TXT record strings static constexpr const char *ServiceTypeFTP = "_ftp._tcp"; - if (!pmDNSPublisher->PublishService (ServiceName, CmDNSPublisher::ServiceTypeFTP, + if (!pmDNSPublisher->PublishService (ServiceName, ServiceTypeFTP, 21, ppText2)) { LOGPANIC ("Cannot publish mdns service"); From 0e484ac6d131e61a25640892bd2414cfe306c231 Mon Sep 17 00:00:00 2001 From: probonopd Date: Sat, 9 Nov 2024 17:16:46 +0100 Subject: [PATCH 34/38] Shorten mDNS interval to 15 seconds --- .github/workflows/build.yml | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/.github/workflows/build.yml b/.github/workflows/build.yml index ff616dd..79de46a 100644 --- a/.github/workflows/build.yml +++ b/.github/workflows/build.yml @@ -19,6 +19,10 @@ jobs: - name: Get specific commits of git submodules run: | sh -ex ./submod.sh + - name: Apply patches + run: | + # https://github.com/rsta2/circle/discussions/427#discussioncomment-11198505 + sed -i -e 's|TTLShort = 120|TTLShort = 15|g' circle-stdlib/libs/circle/net/mdnspublisher.h - name: Install toolchains run: | set -ex From f7e7556b454718107bc28ae3681cd374afa4054d Mon Sep 17 00:00:00 2001 From: probonopd Date: Sat, 9 Nov 2024 17:26:01 +0100 Subject: [PATCH 35/38] Patch include/circle/net/mdnspublisher.h --- .github/workflows/build.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/build.yml b/.github/workflows/build.yml index 79de46a..a960de1 100644 --- a/.github/workflows/build.yml +++ b/.github/workflows/build.yml @@ -22,7 +22,7 @@ jobs: - name: Apply patches run: | # https://github.com/rsta2/circle/discussions/427#discussioncomment-11198505 - sed -i -e 's|TTLShort = 120|TTLShort = 15|g' circle-stdlib/libs/circle/net/mdnspublisher.h + sed -i -e 's|TTLShort = 120|TTLShort = 15|g' circle-stdlib/libs/circle/include/circle/net/mdnspublisher.h - name: Install toolchains run: | set -ex From bf3a93b7730903a859efcc69ac288c41f56c1b6e Mon Sep 17 00:00:00 2001 From: probonopd Date: Sat, 9 Nov 2024 18:00:39 +0100 Subject: [PATCH 36/38] Put git hash in startup message --- .github/workflows/build.yml | 2 ++ 1 file changed, 2 insertions(+) diff --git a/.github/workflows/build.yml b/.github/workflows/build.yml index a960de1..f88831d 100644 --- a/.github/workflows/build.yml +++ b/.github/workflows/build.yml @@ -21,6 +21,8 @@ jobs: sh -ex ./submod.sh - name: Apply patches run: | + # Put git hash in startup message + sed -i "s/Loading.../$(git rev-parse --short=7 HEAD)/g" src/userinterface.c # https://github.com/rsta2/circle/discussions/427#discussioncomment-11198505 sed -i -e 's|TTLShort = 120|TTLShort = 15|g' circle-stdlib/libs/circle/include/circle/net/mdnspublisher.h - name: Install toolchains From 3e64e44152c2b33690f324fbe6e0f21303fea720 Mon Sep 17 00:00:00 2001 From: probonopd Date: Sat, 9 Nov 2024 18:04:03 +0100 Subject: [PATCH 37/38] Patch src/userinterface.cpp --- .github/workflows/build.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/build.yml b/.github/workflows/build.yml index f88831d..20ca9e8 100644 --- a/.github/workflows/build.yml +++ b/.github/workflows/build.yml @@ -22,7 +22,7 @@ jobs: - name: Apply patches run: | # Put git hash in startup message - sed -i "s/Loading.../$(git rev-parse --short=7 HEAD)/g" src/userinterface.c + sed -i "s/Loading.../$(git rev-parse --short=7 HEAD)/g" src/userinterface.cpp # https://github.com/rsta2/circle/discussions/427#discussioncomment-11198505 sed -i -e 's|TTLShort = 120|TTLShort = 15|g' circle-stdlib/libs/circle/include/circle/net/mdnspublisher.h - name: Install toolchains From 29b5d05bd4452ec2b9beacc5060d85c81492ac6c Mon Sep 17 00:00:00 2001 From: probonopd Date: Sat, 9 Nov 2024 18:34:18 +0100 Subject: [PATCH 38/38] Put build date into startup message --- .github/workflows/build.yml | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/.github/workflows/build.yml b/.github/workflows/build.yml index 20ca9e8..c819939 100644 --- a/.github/workflows/build.yml +++ b/.github/workflows/build.yml @@ -22,7 +22,7 @@ jobs: - name: Apply patches run: | # Put git hash in startup message - sed -i "s/Loading.../$(git rev-parse --short=7 HEAD)/g" src/userinterface.cpp + sed -i "s/Loading.../$(date +%Y%m%d)-$(git rev-parse --short HEAD)/g" src/userinterface.cpp # https://github.com/rsta2/circle/discussions/427#discussioncomment-11198505 sed -i -e 's|TTLShort = 120|TTLShort = 15|g' circle-stdlib/libs/circle/include/circle/net/mdnspublisher.h - name: Install toolchains @@ -91,7 +91,7 @@ jobs: git clone https://github.com/Banana71/Soundplantage --depth 1 # depth 1 means only the latest commit cp -r ./Soundplantage/performance ./Soundplantage/*.pdf ./sdcard/ cd sdcard - zip -r ../MiniDexed_$GITHUB_RUN_NUMBER_$(date +%Y-%m-%d)-$(git rev-parse --short HEAD).zip * + zip -r ../MiniDexed_$GITHUB_RUN_NUMBER_$(date +%Y%m%d)-$(git rev-parse --short HEAD).zip * echo "artifactName=MiniDexed_$GITHUB_RUN_NUMBER_$(date +%Y-%m-%d)-$(git rev-parse --short HEAD)" >> $GITHUB_ENV cd - - uses: actions/upload-artifact@v3