|
|
|
@ -21,6 +21,9 @@ |
|
|
|
|
Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA |
|
|
|
|
*/ |
|
|
|
|
|
|
|
|
|
#include "TeensyTimerTool.h" |
|
|
|
|
using namespace TeensyTimerTool; |
|
|
|
|
|
|
|
|
|
#include <limits.h> |
|
|
|
|
#include "config.h" |
|
|
|
|
#include <Audio.h> |
|
|
|
@ -325,6 +328,7 @@ char g_voice_name[NUM_DEXED][VOICE_NAME_LEN]; |
|
|
|
|
char g_bank_name[NUM_DEXED][BANK_NAME_LEN]; |
|
|
|
|
char receive_bank_filename[FILENAME_LEN]; |
|
|
|
|
uint8_t selected_instance_id = 0; |
|
|
|
|
uint8_t seq_UI_last_step = 0; |
|
|
|
|
#ifdef TEENSY4 |
|
|
|
|
#if NUM_DEXED>1 |
|
|
|
|
int8_t midi_decay[NUM_DEXED] = { -1, -1}; |
|
|
|
@ -358,6 +362,8 @@ extern LCDMenuLib2 LCDML; |
|
|
|
|
#endif |
|
|
|
|
|
|
|
|
|
extern void getNoteName(char* noteName, uint8_t noteNumber); |
|
|
|
|
PeriodicTimer timer1; |
|
|
|
|
extern char seq_chord_names[6][4]; |
|
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
|
SETUP |
|
|
|
@ -631,10 +637,13 @@ void setup() |
|
|
|
|
#endif |
|
|
|
|
|
|
|
|
|
LCDML.OTHER_jumpToFunc(UI_func_voice_select); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void loop() |
|
|
|
|
{ |
|
|
|
|
timer1.begin(sequencer, 74'000,false); |
|
|
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void loop() |
|
|
|
|
{ |
|
|
|
|
// MIDI input handling
|
|
|
|
|
check_midi_devices(); |
|
|
|
|
|
|
|
|
@ -642,13 +651,41 @@ void loop() |
|
|
|
|
ENCODER[ENC_L].update(); |
|
|
|
|
ENCODER[ENC_R].update(); |
|
|
|
|
|
|
|
|
|
if (seq_running) { |
|
|
|
|
sequencer(); |
|
|
|
|
} |
|
|
|
|
#ifdef ENABLE_LCD_UI |
|
|
|
|
LCDML.loop(); |
|
|
|
|
#endif |
|
|
|
|
|
|
|
|
|
if (seq_running) |
|
|
|
|
{ |
|
|
|
|
if (seq_step != seq_UI_last_step) |
|
|
|
|
{ |
|
|
|
|
seq_UI_last_step=seq_step; |
|
|
|
|
if (LCDML.FUNC_getID() == LCDML.OTHER_getIDFromFunction(UI_func_sequencer)) //is in UI of Sequencer
|
|
|
|
|
{ |
|
|
|
|
lcd.setCursor(seq_step, 1); |
|
|
|
|
lcd.print("X"); |
|
|
|
|
if (seq_step == 0) |
|
|
|
|
{ |
|
|
|
|
lcd.setCursor(15, 1); |
|
|
|
|
lcd.print(seq_find_shortname(15)[0]); |
|
|
|
|
} |
|
|
|
|
else |
|
|
|
|
{ |
|
|
|
|
lcd.setCursor(seq_step - 1, 1); |
|
|
|
|
lcd.print(seq_find_shortname(seq_step - 1)[0]); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
} |
|
|
|
|
else if (LCDML.FUNC_getID() == LCDML.OTHER_getIDFromFunction(UI_func_arpeggio)) //is in UI of Arpeggiator
|
|
|
|
|
{ |
|
|
|
|
lcd.setCursor(7, 0); |
|
|
|
|
lcd.print( seq_chord_names[arp_chord - 200][0]); |
|
|
|
|
lcd.print( seq_chord_names[arp_chord - 200][1]); |
|
|
|
|
lcd.print( seq_chord_names[arp_chord - 200][2]); |
|
|
|
|
lcd.print( seq_chord_names[arp_chord - 200][3]); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
// CONTROL-RATE-EVENT-HANDLING
|
|
|
|
|
if (control_rate > CONTROL_RATE_MS) |
|
|
|
|
{ |
|
|
|
@ -726,13 +763,13 @@ void loop() |
|
|
|
|
show_cpu_and_mem_usage(); |
|
|
|
|
} |
|
|
|
|
#endif |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/******************************************************************************
|
|
|
|
|
/******************************************************************************
|
|
|
|
|
MIDI MESSAGE HANDLER |
|
|
|
|
******************************************************************************/ |
|
|
|
|
void handleNoteOn(byte inChannel, byte inNumber, byte inVelocity) |
|
|
|
|
{ |
|
|
|
|
void handleNoteOn(byte inChannel, byte inNumber, byte inVelocity) |
|
|
|
|
{ |
|
|
|
|
// Check for MicroDexed
|
|
|
|
|
for (uint8_t instance_id = 0; instance_id < NUM_DEXED; instance_id++) |
|
|
|
|
{ |
|
|
|
@ -819,11 +856,11 @@ void handleNoteOn(byte inChannel, byte inNumber, byte inVelocity) |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
#endif |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
#if NUM_DRUMS > 0 |
|
|
|
|
uint8_t drum_get_slot(uint8_t dt) |
|
|
|
|
{ |
|
|
|
|
uint8_t drum_get_slot(uint8_t dt) |
|
|
|
|
{ |
|
|
|
|
for (uint8_t i = 0; i < NUM_DRUMS; i++) |
|
|
|
|
{ |
|
|
|
|
if (!Drum[i]->isPlaying()) |
|
|
|
@ -850,11 +887,11 @@ uint8_t drum_get_slot(uint8_t dt) |
|
|
|
|
drum_type[drum_counter % 4] = dt; |
|
|
|
|
drum_counter++; |
|
|
|
|
return (drum_counter - 1 % 4); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
#endif |
|
|
|
|
|
|
|
|
|
void handleNoteOff(byte inChannel, byte inNumber, byte inVelocity) |
|
|
|
|
{ |
|
|
|
|
void handleNoteOff(byte inChannel, byte inNumber, byte inVelocity) |
|
|
|
|
{ |
|
|
|
|
for (uint8_t instance_id = 0; instance_id < NUM_DEXED; instance_id++) |
|
|
|
|
{ |
|
|
|
|
if (checkMidiChannel(inChannel, instance_id)) |
|
|
|
@ -879,10 +916,10 @@ void handleNoteOff(byte inChannel, byte inNumber, byte inVelocity) |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void handleControlChange(byte inChannel, byte inCtrl, byte inValue) |
|
|
|
|
{ |
|
|
|
|
void handleControlChange(byte inChannel, byte inCtrl, byte inValue) |
|
|
|
|
{ |
|
|
|
|
inCtrl = constrain(inCtrl, 0, 127); |
|
|
|
|
inValue = constrain(inValue, 0, 127); |
|
|
|
|
|
|
|
|
@ -1095,10 +1132,10 @@ void handleControlChange(byte inChannel, byte inCtrl, byte inValue) |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void handleAfterTouch(byte inChannel, byte inPressure) |
|
|
|
|
{ |
|
|
|
|
void handleAfterTouch(byte inChannel, byte inPressure) |
|
|
|
|
{ |
|
|
|
|
for (uint8_t instance_id = 0; instance_id < NUM_DEXED; instance_id++) |
|
|
|
|
{ |
|
|
|
|
if (checkMidiChannel(inChannel, instance_id)) |
|
|
|
@ -1107,10 +1144,10 @@ void handleAfterTouch(byte inChannel, byte inPressure) |
|
|
|
|
MicroDexed[instance_id]->ControllersRefresh(); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void handlePitchBend(byte inChannel, int inPitch) |
|
|
|
|
{ |
|
|
|
|
void handlePitchBend(byte inChannel, int inPitch) |
|
|
|
|
{ |
|
|
|
|
for (uint8_t instance_id = 0; instance_id < NUM_DEXED; instance_id++) |
|
|
|
|
{ |
|
|
|
|
if (checkMidiChannel(inChannel, instance_id)) |
|
|
|
@ -1118,10 +1155,10 @@ void handlePitchBend(byte inChannel, int inPitch) |
|
|
|
|
MicroDexed[instance_id]->setPitchbend(inPitch); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void handleProgramChange(byte inChannel, byte inProgram) |
|
|
|
|
{ |
|
|
|
|
void handleProgramChange(byte inChannel, byte inProgram) |
|
|
|
|
{ |
|
|
|
|
for (uint8_t instance_id = 0; instance_id < NUM_DEXED; instance_id++) |
|
|
|
|
{ |
|
|
|
|
if (checkMidiChannel(inChannel, instance_id)) |
|
|
|
@ -1141,10 +1178,10 @@ void handleProgramChange(byte inChannel, byte inProgram) |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void handleSystemExclusive(byte * sysex, uint len) |
|
|
|
|
{ |
|
|
|
|
void handleSystemExclusive(byte * sysex, uint len) |
|
|
|
|
{ |
|
|
|
|
for (uint8_t instance_id = 0; instance_id < NUM_DEXED; instance_id++) |
|
|
|
|
{ |
|
|
|
|
if (!checkMidiChannel((sysex[2] & 0x0f) + 1 , instance_id)) |
|
|
|
@ -1543,30 +1580,30 @@ void handleSystemExclusive(byte * sysex, uint len) |
|
|
|
|
Serial.println(F("E: SysEx parameter length wrong.")); |
|
|
|
|
#endif |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void handleTimeCodeQuarterFrame(byte data) |
|
|
|
|
{ |
|
|
|
|
void handleTimeCodeQuarterFrame(byte data) |
|
|
|
|
{ |
|
|
|
|
; |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void handleAfterTouchPoly(byte inChannel, byte inNumber, byte inVelocity) |
|
|
|
|
{ |
|
|
|
|
void handleAfterTouchPoly(byte inChannel, byte inNumber, byte inVelocity) |
|
|
|
|
{ |
|
|
|
|
; |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void handleSongSelect(byte inSong) |
|
|
|
|
{ |
|
|
|
|
void handleSongSelect(byte inSong) |
|
|
|
|
{ |
|
|
|
|
; |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void handleTuneRequest(void) |
|
|
|
|
{ |
|
|
|
|
void handleTuneRequest(void) |
|
|
|
|
{ |
|
|
|
|
; |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void handleClock(void) |
|
|
|
|
{ |
|
|
|
|
void handleClock(void) |
|
|
|
|
{ |
|
|
|
|
if (midi_bpm_counter % 24 == 0) |
|
|
|
|
{ |
|
|
|
|
midi_bpm = (60000.0f / float(midi_bpm_timer) + 0.5); |
|
|
|
@ -1617,32 +1654,32 @@ void handleClock(void) |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
midi_bpm_counter++; |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void handleStart(void) |
|
|
|
|
{ |
|
|
|
|
void handleStart(void) |
|
|
|
|
{ |
|
|
|
|
midi_bpm_timer = 0; |
|
|
|
|
midi_bpm_counter = 0; |
|
|
|
|
_midi_bpm = -1; |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void handleContinue(void) |
|
|
|
|
{ |
|
|
|
|
void handleContinue(void) |
|
|
|
|
{ |
|
|
|
|
; |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void handleStop(void) |
|
|
|
|
{ |
|
|
|
|
void handleStop(void) |
|
|
|
|
{ |
|
|
|
|
; |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void handleActiveSensing(void) |
|
|
|
|
{ |
|
|
|
|
void handleActiveSensing(void) |
|
|
|
|
{ |
|
|
|
|
; |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void handleSystemReset(void) |
|
|
|
|
{ |
|
|
|
|
void handleSystemReset(void) |
|
|
|
|
{ |
|
|
|
|
for (uint8_t instance_id = 0; instance_id < NUM_DEXED; instance_id++) |
|
|
|
|
{ |
|
|
|
|
#ifdef DEBUG |
|
|
|
@ -1652,13 +1689,13 @@ void handleSystemReset(void) |
|
|
|
|
MicroDexed[instance_id]->panic(); |
|
|
|
|
MicroDexed[instance_id]->resetControllers(); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/******************************************************************************
|
|
|
|
|
/******************************************************************************
|
|
|
|
|
MIDI HELPER |
|
|
|
|
******************************************************************************/ |
|
|
|
|
bool checkMidiChannel(byte inChannel, uint8_t instance_id) |
|
|
|
|
{ |
|
|
|
|
bool checkMidiChannel(byte inChannel, uint8_t instance_id) |
|
|
|
|
{ |
|
|
|
|
// check for MIDI channel
|
|
|
|
|
if (configuration.dexed[instance_id].midi_channel == MIDI_CHANNEL_OMNI) |
|
|
|
|
{ |
|
|
|
@ -1678,10 +1715,10 @@ bool checkMidiChannel(byte inChannel, uint8_t instance_id) |
|
|
|
|
return (false); |
|
|
|
|
} |
|
|
|
|
return (true); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void init_MIDI_send_CC(void) |
|
|
|
|
{ |
|
|
|
|
void init_MIDI_send_CC(void) |
|
|
|
|
{ |
|
|
|
|
#ifdef DEBUG |
|
|
|
|
Serial.println("init_MIDI_send_CC():"); |
|
|
|
|
#endif |
|
|
|
@ -1695,14 +1732,14 @@ void init_MIDI_send_CC(void) |
|
|
|
|
MD_sendControlChange(configuration.dexed[selected_instance_id].midi_channel, 105, configuration.fx.delay_time[selected_instance_id]); |
|
|
|
|
MD_sendControlChange(configuration.dexed[selected_instance_id].midi_channel, 106, configuration.fx.delay_feedback[selected_instance_id]); |
|
|
|
|
MD_sendControlChange(configuration.dexed[selected_instance_id].midi_channel, 107, configuration.fx.delay_level[selected_instance_id]); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/******************************************************************************
|
|
|
|
|
/******************************************************************************
|
|
|
|
|
VOLUME HELPER |
|
|
|
|
******************************************************************************/ |
|
|
|
|
|
|
|
|
|
void set_volume(uint8_t v, uint8_t m) |
|
|
|
|
{ |
|
|
|
|
void set_volume(uint8_t v, uint8_t m) |
|
|
|
|
{ |
|
|
|
|
configuration.sys.vol = v; |
|
|
|
|
|
|
|
|
|
if (configuration.sys.vol > 100) |
|
|
|
@ -1743,14 +1780,14 @@ void set_volume(uint8_t v, uint8_t m) |
|
|
|
|
mono2stereo[instance_id]->panorama(mapfloat(PANORAMA_MIN, PANORAMA_MIN, PANORAMA_MAX, -1.0, 1.0)); |
|
|
|
|
break; |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/******************************************************************************
|
|
|
|
|
/******************************************************************************
|
|
|
|
|
EEPROM HELPER |
|
|
|
|
******************************************************************************/ |
|
|
|
|
|
|
|
|
|
void initial_values_from_eeprom(bool init) |
|
|
|
|
{ |
|
|
|
|
void initial_values_from_eeprom(bool init) |
|
|
|
|
{ |
|
|
|
|
uint16_t _m_; |
|
|
|
|
|
|
|
|
|
if (init == true) |
|
|
|
@ -1797,28 +1834,28 @@ void initial_values_from_eeprom(bool init) |
|
|
|
|
#ifdef DEBUG |
|
|
|
|
show_configuration(); |
|
|
|
|
#endif |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void check_configuration(void) |
|
|
|
|
{ |
|
|
|
|
void check_configuration(void) |
|
|
|
|
{ |
|
|
|
|
check_configuration_sys(); |
|
|
|
|
check_configuration_fx(); |
|
|
|
|
check_configuration_performance(); |
|
|
|
|
for (uint8_t instance_id = 0; instance_id < NUM_DEXED; instance_id++) |
|
|
|
|
check_configuration_dexed(instance_id); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void check_configuration_sys(void) |
|
|
|
|
{ |
|
|
|
|
void check_configuration_sys(void) |
|
|
|
|
{ |
|
|
|
|
configuration.sys.instances = constrain(configuration.sys.instances, INSTANCES_MIN, INSTANCES_MAX); |
|
|
|
|
configuration.sys.vol = constrain(configuration.sys.vol, VOLUME_MIN, VOLUME_MAX); |
|
|
|
|
configuration.sys.mono = constrain(configuration.sys.mono, MONO_MIN, MONO_MAX); |
|
|
|
|
configuration.sys.soft_midi_thru = constrain(configuration.sys.soft_midi_thru, SOFT_MIDI_THRU_MIN, SOFT_MIDI_THRU_MAX); |
|
|
|
|
configuration.sys.performance_number = constrain(configuration.sys.performance_number, PERFORMANCE_NUM_MIN, PERFORMANCE_NUM_MAX); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void check_configuration_fx(void) |
|
|
|
|
{ |
|
|
|
|
void check_configuration_fx(void) |
|
|
|
|
{ |
|
|
|
|
#ifdef USE_PLATEREVERB |
|
|
|
|
configuration.fx.reverb_lowpass = constrain(configuration.fx.reverb_lowpass, REVERB_LOWPASS_MIN, REVERB_LOWPASS_MAX); |
|
|
|
|
configuration.fx.reverb_lodamp = constrain(configuration.fx.reverb_lodamp, REVERB_LODAMP_MIN, REVERB_LODAMP_MAX); |
|
|
|
@ -1844,10 +1881,10 @@ void check_configuration_fx(void) |
|
|
|
|
configuration.fx.delay_sync[instance_id] = constrain(configuration.fx.delay_sync[instance_id], DELAY_SYNC_MIN, DELAY_SYNC_MAX); |
|
|
|
|
configuration.fx.reverb_send[instance_id] = constrain(configuration.fx.reverb_send[instance_id], REVERB_SEND_MIN, REVERB_SEND_MAX); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void check_configuration_performance(void) |
|
|
|
|
{ |
|
|
|
|
void check_configuration_performance(void) |
|
|
|
|
{ |
|
|
|
|
configuration.performance.fx_number = constrain(configuration.performance.fx_number, FX_NUM_MIN, FX_NUM_MAX); |
|
|
|
|
|
|
|
|
|
for (uint8_t instance_id = 0; instance_id < NUM_DEXED; instance_id++) |
|
|
|
@ -1856,10 +1893,10 @@ void check_configuration_performance(void) |
|
|
|
|
configuration.performance.voice[instance_id] = constrain(configuration.performance.voice[instance_id], 0, MAX_VOICES - 1); |
|
|
|
|
configuration.performance.voiceconfig_number[instance_id] = constrain(configuration.performance.voiceconfig_number[instance_id], VOICECONFIG_NUM_MIN, VOICECONFIG_NUM_MAX); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void check_configuration_dexed(uint8_t instance_id) |
|
|
|
|
{ |
|
|
|
|
void check_configuration_dexed(uint8_t instance_id) |
|
|
|
|
{ |
|
|
|
|
configuration.dexed[instance_id].midi_channel = constrain(configuration.dexed[instance_id].midi_channel, MIDI_CHANNEL_MIN, MIDI_CHANNEL_MAX); |
|
|
|
|
configuration.dexed[instance_id].lowest_note = constrain(configuration.dexed[instance_id].lowest_note, INSTANCE_LOWEST_NOTE_MIN, INSTANCE_LOWEST_NOTE_MAX); |
|
|
|
|
configuration.dexed[instance_id].highest_note = constrain(configuration.dexed[instance_id].highest_note, INSTANCE_HIGHEST_NOTE_MIN, INSTANCE_HIGHEST_NOTE_MAX); |
|
|
|
@ -1889,10 +1926,10 @@ void check_configuration_dexed(uint8_t instance_id) |
|
|
|
|
configuration.dexed[instance_id].portamento_glissando = constrain(configuration.dexed[instance_id].portamento_glissando, PORTAMENTO_GLISSANDO_MIN, PORTAMENTO_GLISSANDO_MAX); |
|
|
|
|
configuration.dexed[instance_id].portamento_time = constrain(configuration.dexed[instance_id].portamento_time, PORTAMENTO_TIME_MIN, PORTAMENTO_TIME_MAX); |
|
|
|
|
configuration.dexed[instance_id].op_enabled = constrain(configuration.dexed[instance_id].op_enabled, OP_ENABLED_MIN, OP_ENABLED_MAX); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void init_configuration(void) |
|
|
|
|
{ |
|
|
|
|
void init_configuration(void) |
|
|
|
|
{ |
|
|
|
|
#ifdef DEBUG |
|
|
|
|
Serial.println(F("INITIALIZING CONFIGURATION")); |
|
|
|
|
#endif |
|
|
|
@ -1976,17 +2013,17 @@ void init_configuration(void) |
|
|
|
|
set_volume(configuration.sys.vol, configuration.sys.mono); |
|
|
|
|
|
|
|
|
|
eeprom_update(); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void eeprom_update(void) |
|
|
|
|
{ |
|
|
|
|
void eeprom_update(void) |
|
|
|
|
{ |
|
|
|
|
uint8_t* c = (uint8_t*)&configuration; |
|
|
|
|
for (uint16_t i = 0; i < sizeof(configuration); i++) |
|
|
|
|
EEPROM.update(EEPROM_START_ADDRESS + i, c[i]); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void eeprom_update_sys(void) |
|
|
|
|
{ |
|
|
|
|
void eeprom_update_sys(void) |
|
|
|
|
{ |
|
|
|
|
uint8_t* c = (uint8_t*)&configuration.sys; |
|
|
|
|
|
|
|
|
|
for (uint16_t i = 0; i < sizeof(configuration.sys); i++) |
|
|
|
@ -1995,16 +2032,16 @@ void eeprom_update_sys(void) |
|
|
|
|
#ifdef DEBUG |
|
|
|
|
Serial.println(F("Updating EEPROM sys.")); |
|
|
|
|
#endif |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
bool eeprom_get_sys(void) |
|
|
|
|
{ |
|
|
|
|
bool eeprom_get_sys(void) |
|
|
|
|
{ |
|
|
|
|
EEPROM.get(EEPROM_START_ADDRESS + offsetof(configuration_s, sys), configuration.sys); |
|
|
|
|
return (true); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void eeprom_update_fx(void) |
|
|
|
|
{ |
|
|
|
|
void eeprom_update_fx(void) |
|
|
|
|
{ |
|
|
|
|
uint8_t* c = (uint8_t*)&configuration.fx; |
|
|
|
|
|
|
|
|
|
for (uint16_t i = 0; i < sizeof(configuration.fx); i++) |
|
|
|
@ -2013,16 +2050,16 @@ void eeprom_update_fx(void) |
|
|
|
|
#ifdef DEBUG |
|
|
|
|
Serial.println(F("Updating EEPROM fx.")); |
|
|
|
|
#endif |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
bool eeprom_get_fx(void) |
|
|
|
|
{ |
|
|
|
|
bool eeprom_get_fx(void) |
|
|
|
|
{ |
|
|
|
|
EEPROM.get(EEPROM_START_ADDRESS + offsetof(configuration_s, fx), configuration.fx); |
|
|
|
|
return (true); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void eeprom_update_dexed(uint8_t instance_id) |
|
|
|
|
{ |
|
|
|
|
void eeprom_update_dexed(uint8_t instance_id) |
|
|
|
|
{ |
|
|
|
|
#if NUM_DEXED == 1 |
|
|
|
|
uint8_t* c = (uint8_t*)&configuration.dexed[0]; |
|
|
|
|
|
|
|
|
@ -2050,10 +2087,10 @@ void eeprom_update_dexed(uint8_t instance_id) |
|
|
|
|
Serial.print(instance_id); |
|
|
|
|
Serial.println(F(").")); |
|
|
|
|
#endif |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
bool eeprom_get_dexed(uint8_t instance_id) |
|
|
|
|
{ |
|
|
|
|
bool eeprom_get_dexed(uint8_t instance_id) |
|
|
|
|
{ |
|
|
|
|
for (uint8_t instance_id = 0; instance_id < MAX_DEXED; instance_id++) |
|
|
|
|
{ |
|
|
|
|
if (instance_id == 0) |
|
|
|
@ -2062,31 +2099,31 @@ bool eeprom_get_dexed(uint8_t instance_id) |
|
|
|
|
EEPROM.get(EEPROM_START_ADDRESS + offsetof(configuration_s, dexed[1]), configuration.dexed[1]); |
|
|
|
|
} |
|
|
|
|
return (true); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void eeprom_update_performance() |
|
|
|
|
{ |
|
|
|
|
void eeprom_update_performance() |
|
|
|
|
{ |
|
|
|
|
EEPROM.put(EEPROM_START_ADDRESS + offsetof(configuration_s, performance), configuration.performance); |
|
|
|
|
#ifdef DEBUG |
|
|
|
|
Serial.println(F("Updating EEPROM performance.")); |
|
|
|
|
#endif |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
bool eeprom_get_performance() |
|
|
|
|
{ |
|
|
|
|
bool eeprom_get_performance() |
|
|
|
|
{ |
|
|
|
|
EEPROM.get(EEPROM_START_ADDRESS + offsetof(configuration_s, performance), configuration.performance); |
|
|
|
|
#ifdef DEBUG |
|
|
|
|
Serial.println(F("Getting EEPROM performance.")); |
|
|
|
|
#endif |
|
|
|
|
return (true); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/******************************************************************************
|
|
|
|
|
/******************************************************************************
|
|
|
|
|
PARAMETER-HELPERS |
|
|
|
|
******************************************************************************/ |
|
|
|
|
******************************************************************************/ |
|
|
|
|
|
|
|
|
|
void set_fx_params(void) |
|
|
|
|
{ |
|
|
|
|
void set_fx_params(void) |
|
|
|
|
{ |
|
|
|
|
#if defined(USE_FX) |
|
|
|
|
for (uint8_t instance_id = 0; instance_id < NUM_DEXED; instance_id++) |
|
|
|
|
{ |
|
|
|
@ -2173,10 +2210,10 @@ void set_fx_params(void) |
|
|
|
|
#endif |
|
|
|
|
|
|
|
|
|
init_MIDI_send_CC(); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void set_voiceconfig_params(uint8_t instance_id) |
|
|
|
|
{ |
|
|
|
|
void set_voiceconfig_params(uint8_t instance_id) |
|
|
|
|
{ |
|
|
|
|
// INIT PEAK MIXER
|
|
|
|
|
microdexed_peak_mixer.gain(instance_id, 1.0); |
|
|
|
|
|
|
|
|
@ -2197,36 +2234,36 @@ void set_voiceconfig_params(uint8_t instance_id) |
|
|
|
|
|
|
|
|
|
// PANORAMA
|
|
|
|
|
mono2stereo[instance_id]->panorama(mapfloat(configuration.dexed[instance_id].pan, PANORAMA_MIN, PANORAMA_MAX, -1.0, 1.0)); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void set_sys_params(void) |
|
|
|
|
{ |
|
|
|
|
void set_sys_params(void) |
|
|
|
|
{ |
|
|
|
|
// set initial volume
|
|
|
|
|
set_volume(configuration.sys.vol, configuration.sys.mono); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/******************************************************************************
|
|
|
|
|
/******************************************************************************
|
|
|
|
|
HELPERS |
|
|
|
|
******************************************************************************/ |
|
|
|
|
|
|
|
|
|
// https://www.reddit.com/r/Teensy/comments/7r19uk/reset_and_reboot_teensy_lc_via_code/
|
|
|
|
|
// https://www.reddit.com/r/Teensy/comments/7r19uk/reset_and_reboot_teensy_lc_via_code/
|
|
|
|
|
#define SCB_AIRCR (*(volatile uint32_t *)0xE000ED0C) // Application Interrupt and Reset Control location
|
|
|
|
|
void _softRestart(void) |
|
|
|
|
{ |
|
|
|
|
void _softRestart(void) |
|
|
|
|
{ |
|
|
|
|
Serial.end(); //clears the serial monitor if used
|
|
|
|
|
SCB_AIRCR = 0x05FA0004; //write value for restart
|
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
float pseudo_log_curve(float value) |
|
|
|
|
{ |
|
|
|
|
float pseudo_log_curve(float value) |
|
|
|
|
{ |
|
|
|
|
//return (mapfloat(_pseudo_log * arm_sin_f32(value), 0.0, _pseudo_log * arm_sin_f32(1.0), 0.0, 1.0));
|
|
|
|
|
//return (1 - sqrt(1 - value * value));
|
|
|
|
|
//return (pow(2, value) - 1);
|
|
|
|
|
return (pow(value, 2.2)); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
uint32_t crc32(byte * calc_start, uint16_t calc_bytes) // base code from https://www.arduino.cc/en/Tutorial/EEPROMCrc
|
|
|
|
|
{ |
|
|
|
|
uint32_t crc32(byte * calc_start, uint16_t calc_bytes) // base code from https://www.arduino.cc/en/Tutorial/EEPROMCrc
|
|
|
|
|
{ |
|
|
|
|
const uint32_t crc_table[16] = |
|
|
|
|
{ |
|
|
|
|
0x00000000, 0x1db71064, 0x3b6e20c8, 0x26d930ac, |
|
|
|
@ -2244,10 +2281,10 @@ uint32_t crc32(byte * calc_start, uint16_t calc_bytes) // base code from https:/ |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
return (crc); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void generate_version_string(char* buffer, uint8_t len) |
|
|
|
|
{ |
|
|
|
|
void generate_version_string(char* buffer, uint8_t len) |
|
|
|
|
{ |
|
|
|
|
char tmp[3]; |
|
|
|
|
|
|
|
|
|
memset(buffer, 0, len); |
|
|
|
@ -2267,21 +2304,21 @@ void generate_version_string(char* buffer, uint8_t len) |
|
|
|
|
itoa (MAX_NOTES, tmp, 10); |
|
|
|
|
strncat(buffer, tmp, 2); |
|
|
|
|
#endif |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
#ifdef DISPLAY_LCD_SPI |
|
|
|
|
void change_disp_sd(bool disp) |
|
|
|
|
{ |
|
|
|
|
void change_disp_sd(bool disp) |
|
|
|
|
{ |
|
|
|
|
if (sd_card > 0) |
|
|
|
|
{ |
|
|
|
|
digitalWrite(sd_card, disp); |
|
|
|
|
digitalWrite(U8X8_CS_PIN, !disp); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
#endif |
|
|
|
|
|
|
|
|
|
uint8_t check_sd_cards(void) |
|
|
|
|
{ |
|
|
|
|
uint8_t check_sd_cards(void) |
|
|
|
|
{ |
|
|
|
|
uint8_t ret = 0; |
|
|
|
|
|
|
|
|
|
memset(sd_string, 0, sizeof(sd_string)); |
|
|
|
@ -2379,10 +2416,10 @@ uint8_t check_sd_cards(void) |
|
|
|
|
#endif |
|
|
|
|
|
|
|
|
|
return (ret); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void check_and_create_directories(void) |
|
|
|
|
{ |
|
|
|
|
void check_and_create_directories(void) |
|
|
|
|
{ |
|
|
|
|
if (sd_card > 0) |
|
|
|
|
{ |
|
|
|
|
uint8_t i; |
|
|
|
@ -2490,14 +2527,14 @@ void check_and_create_directories(void) |
|
|
|
|
Serial.println(F("No SD card for directory check available.")); |
|
|
|
|
#endif |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/******************************************************************************
|
|
|
|
|
/******************************************************************************
|
|
|
|
|
DEBUG HELPER |
|
|
|
|
******************************************************************************/ |
|
|
|
|
#if defined (DEBUG) && defined (SHOW_CPU_LOAD_MSEC) |
|
|
|
|
void show_cpu_and_mem_usage(void) |
|
|
|
|
{ |
|
|
|
|
void show_cpu_and_mem_usage(void) |
|
|
|
|
{ |
|
|
|
|
uint32_t sum_xrun = 0; |
|
|
|
|
uint16_t sum_render_time_max = 0; |
|
|
|
|
|
|
|
|
@ -2567,12 +2604,12 @@ void show_cpu_and_mem_usage(void) |
|
|
|
|
#endif |
|
|
|
|
AudioProcessorUsageMaxReset(); |
|
|
|
|
AudioMemoryUsageMaxReset(); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
#endif |
|
|
|
|
|
|
|
|
|
#ifdef DEBUG |
|
|
|
|
void show_configuration(void) |
|
|
|
|
{ |
|
|
|
|
void show_configuration(void) |
|
|
|
|
{ |
|
|
|
|
Serial.println(); |
|
|
|
|
Serial.println(F("CONFIGURATION:")); |
|
|
|
|
Serial.println(F("System")); |
|
|
|
@ -2648,10 +2685,10 @@ void show_configuration(void) |
|
|
|
|
|
|
|
|
|
Serial.println(); |
|
|
|
|
Serial.flush(); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void show_patch(uint8_t instance_id) |
|
|
|
|
{ |
|
|
|
|
void show_patch(uint8_t instance_id) |
|
|
|
|
{ |
|
|
|
|
char vn[VOICE_NAME_LEN]; |
|
|
|
|
|
|
|
|
|
Serial.print(F("INSTANCE ")); |
|
|
|
@ -2756,30 +2793,30 @@ void show_patch(uint8_t instance_id) |
|
|
|
|
Serial.println(F(" |")); |
|
|
|
|
Serial.println(F("+=========+=========+=========+=========+==========+==========+=========+==================================+")); |
|
|
|
|
Serial.println(F("+==========================================================================================================+")); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void SerialPrintFormatInt3(uint8_t num) |
|
|
|
|
{ |
|
|
|
|
void SerialPrintFormatInt3(uint8_t num) |
|
|
|
|
{ |
|
|
|
|
char buf[4]; |
|
|
|
|
sprintf(buf, "%3d", num); |
|
|
|
|
Serial.print(buf); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
#ifdef TEENSY3_6 |
|
|
|
|
/* From: https://forum.pjrc.com/threads/33443-How-to-display-free-ram */ |
|
|
|
|
extern "C" char* sbrk(int incr); |
|
|
|
|
uint32_t FreeMem(void) |
|
|
|
|
{ |
|
|
|
|
/* From: https://forum.pjrc.com/threads/33443-How-to-display-free-ram */ |
|
|
|
|
extern "C" char* sbrk(int incr); |
|
|
|
|
uint32_t FreeMem(void) |
|
|
|
|
{ |
|
|
|
|
char top; |
|
|
|
|
return &top - reinterpret_cast<char*>(sbrk(0)); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
#else |
|
|
|
|
/* From: https://forum.pjrc.com/threads/33443-How-to-display-free-ram */ |
|
|
|
|
extern unsigned long _heap_end; |
|
|
|
|
extern char *__brkval; |
|
|
|
|
int FreeMem(void) |
|
|
|
|
{ |
|
|
|
|
/* From: https://forum.pjrc.com/threads/33443-How-to-display-free-ram */ |
|
|
|
|
extern unsigned long _heap_end; |
|
|
|
|
extern char *__brkval; |
|
|
|
|
int FreeMem(void) |
|
|
|
|
{ |
|
|
|
|
return (char *)&_heap_end - __brkval; |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
#endif |
|
|
|
|
#endif |
|
|
|
|