// 2 Operator FM Synth firmware for Sitka Instruments WS-1.0 // by Oleksiy Hrachov // // Code is partly based on Knob_LightLevel_x2_FMSynth example from Mozzi Library // // Although the code designed to work on Sitka Instruments WS-1.0 synth, it should // be pretty easy to adapt to run on other arduino/mozzi-based setups. // Edit config.h to match your setup. // // This code is licenced under GPL v3 or later //ToDo: //optimize MIDI input (maybe neohwserial and this parser might help https://github.com/eclab/grains/tree/main/midi) //add ramp LFO //add uneven rations (like 3:2, 3:5) //rework LFO/Intensity so when LFO rate is 0, intensity doesn't depend on its phase #include #include #include #include #include #include #include #include #include #include #include #include #include "config.h" IntMap kMapCarrierNote( 0, 4095, 24 * pitchSubSteps, 84 * pitchSubSteps ); IntMap kMapIntensity( 0, 1023, 10, 350 ); IntMap kMapHarmonics( 0, 1023, 1, 40 ); IntMap kMapLFOSpeed( 0, 1023, 1, 10000 ); IntMap kMapAttack( 0, 1023, 0, 80 ); IntMap kMapDecayRelease( 0, 1023, 8, 160 ); IntMap kMapSustain( 0, 1023, 0, 255 ); Oscil aCarrier(SIN2048_DATA); Oscil aModulator(SIN2048_DATA); Oscil kSineLFO(SIN2048_DATA); Oscil kSquareLFO(SQUARE_NO_ALIAS_2048_DATA); Oscil kSawLFO(SAW2048_DATA); Oscil kNoiseLFO(WHITENOISE8192_DATA); ADSR envelope; Smooth aSmoothIntensity(0.95f); OverSample kOverSamplePitch; //Global variables byte gain; bool MIDINotePlaying; bool gateIsHigh = false; float carrierFreq; long FMIntensity; // carries control info from updateControl to updateAudio void MIDINoteOn(byte channel, byte note, byte velocity) { carrierFreq = mtof((int) note); envelope.noteOn(); MIDINotePlaying = true; digitalWrite(LED, LOW); } void MIDINoteOff(byte channel, byte note, byte velocity) { envelope.noteOff(); digitalWrite(LED, HIGH); } long softClip(long input) { int threshold = 2048; if (input < -threshold) { return -threshold + (input + threshold) / 4; } else if (input > threshold) { return threshold + (input - threshold) / 4; } else { return input; } } void setup(){ pinMode(LED, OUTPUT); pinMode(GateIn, INPUT_PULLUP); pinMode(EnvSwitch, INPUT_PULLUP); pinMode(DroneSwitch, INPUT_PULLUP); MIDI.setHandleNoteOn(MIDINoteOn); MIDI.setHandleNoteOff(MIDINoteOff); MIDI.setHandleClock(nullptr); MIDI.setHandleStart(nullptr); MIDI.setHandleStop(nullptr); MIDI.setHandleContinue(nullptr); MIDI.setHandleControlChange(nullptr); MIDI.begin(MIDI_CHANNEL); MIDI.turnThruOff(); startMozzi(); envelope.setAttackLevel(255); digitalWrite(LED, HIGH); } void updateControl(){ for(int i=0; i<10; i++) { // Hacky way to drain the buffer faster MIDI.read(); } //Get Control Values int CVInVal = mozziAnalogRead(CVIn); int knob1Val = mozziAnalogRead(Knob1); int knob2Val = mozziAnalogRead(Knob2); int knob3Val = mozziAnalogRead(Knob3); int knob4Val = mozziAnalogRead(Knob4); int knobAVal = mozziAnalogRead(KnobA); int knobDRVal = mozziAnalogRead(KnobDR); int knobSVal = mozziAnalogRead(KnobS); bool droneSwitchVal = digitalRead(DroneSwitch); bool envSwitchVal = digitalRead(EnvSwitch); bool gateInVal = !digitalRead(GateIn); //Remap the values and assign to parameter int intensity = kMapIntensity(knob1Val); int harmonics = kMapHarmonics(knob2Val); int LFOSpeed = kMapLFOSpeed(knob3Val); float expLFOSpeed = (float) LFOSpeed * LFOSpeed / 400000; float modSpeed = expLFOSpeed; //Set pitch and play notes on trigger if (!MIDINotePlaying) { if (oversamplingEnabled) { int oversampledCVInVal = kOverSamplePitch.next(CVInVal); carrierFreq = mtof((float) kMapCarrierNote(oversampledCVInVal) / pitchSubSteps); } else { carrierFreq = mtof((float) kMapCarrierNote(CVInVal << 2) / pitchSubSteps); } digitalWrite(LED, !gateInVal); if (gateInVal && !gateIsHigh) { gateIsHigh = true; envelope.noteOn(); } else if (!gateInVal && gateIsHigh) { gateIsHigh = false; envelope.noteOff(); } } else if (MIDINotePlaying && !envelope.playing()) { MIDINotePlaying = false; } //Update Envelope Settings int attackTime = kMapAttack(knobAVal); int decayReleaseTime = kMapDecayRelease(knobDRVal); int sustainLevel = kMapSustain(knobSVal); envelope.setDecayLevel(sustainLevel); envelope.setTimes(attackTime, decayReleaseTime, 30000, decayReleaseTime); //30000 is so the note will sustain 30 seconds unless a noteOff comes //Calculate the modulation frequency int FMmodFreq = carrierFreq * harmonics; //Set oscillator frequencies aCarrier.setFreq(carrierFreq); aModulator.setFreq(FMmodFreq); kSineLFO.setFreq(modSpeed); kSquareLFO.setFreq(modSpeed); kSawLFO.setFreq(modSpeed); kNoiseLFO.setFreq(modSpeed/4096); envelope.update(); int sineLFOLevel; int squareLFOLevel; int sawLFOLevel; int noiseLFOLevel; if (knob4Val < 255) { sineLFOLevel = 255 - knob4Val; squareLFOLevel = knob4Val; sawLFOLevel = 0; noiseLFOLevel = 0; } else if (knob4Val < 511) { sineLFOLevel = 0; squareLFOLevel = 511 - knob4Val; sawLFOLevel = knob4Val - 255; noiseLFOLevel = 0; } else if (knob4Val < 767) { sineLFOLevel = 0; squareLFOLevel = 0; sawLFOLevel = 767 - knob4Val; noiseLFOLevel = knob4Val - 511; } else { sineLFOLevel = 0; squareLFOLevel = 0; sawLFOLevel = 0; noiseLFOLevel = 255; } int shapedLFO = (kSineLFO.next() * sineLFOLevel + kSquareLFO.next() * squareLFOLevel + kSawLFO.next() * sawLFOLevel + kNoiseLFO.next() * noiseLFOLevel)>>8; int env = envelope.next(); int intensityEnv; if(envSwitchVal) { intensityEnv = env*3; } else { intensityEnv = 1; } FMIntensity = ((long)(intensity + intensityEnv) * (shapedLFO+128))>>8; //(shapedLFO+128))>>16; //(kSineLFO.next()+128))>>8; if(!droneSwitchVal) { gain = env; } else { gain = 255; } } AudioOutput updateAudio(){ long modulation = aSmoothIntensity.next(FMIntensity) * aModulator.next(); long signal = (aCarrier.phMod(modulation) * gain) >> 8; //envelope signal = softClip((signal * (127 + driveAmount)) >> 8); //overdrive return MonoOutput::from8Bit(signal); } void loop(){ audioHook(); }