Mercurial > hg > beaglert
view projects/d-box/OscillatorBank.cpp @ 45:579c86316008 newapi
Major API overhaul. Moved to a single data structure for handling render functions. Functionally, generally similar except for scheduling within PRU loop function, which now uses interrupts from the PRU rather than polling. This requires an updated kernel.
author | andrewm |
---|---|
date | Thu, 28 May 2015 14:35:55 -0400 |
parents | 8a575ba3ab52 |
children |
line wrap: on
line source
/* * OscillatorBank.cpp * * Created on: May 23, 2014 * Author: Victor Zappi and Andrew McPherson */ /* * There is a problem with name consistency between this class and the Parser class in spear_parser files. * There, a "frame" is each of the time values where partials are sampled, while a "hop" is the actual jump between frames [calculated in samples] * Here, "hop" is used with the meaning of "frame", while "frame" became the local frame of a partial * * example * * frames: 0 1 2 * p0 p0_0 p0_1 * p1 p1_0 p1_1 p1_2 * p2 p2_0 p2_1 * * In this case: * in Parser there are 2 hops, 3 total frames and the 3 partials have respectively 2, 3 and 2 local frames * * here there are 3 total hops [the concept of jumps is discarded, cos not in use] and the 3 partials have respectively 2, 3 and 2 frames * * This must be fixed */ // TODO: fix hop-frame name consistency #include <stdlib.h> #include "OscillatorBank.h" OscillatorBank::OscillatorBank() { loaded = false; } OscillatorBank::OscillatorBank(string filename, int hopsize, int samplerate) { loaded = false; loadFile(filename.c_str(), hopsize, samplerate); } OscillatorBank::OscillatorBank(char *filename, int hopsize, int samplerate) { loaded = false; loadFile(filename, hopsize, samplerate); } OscillatorBank::~OscillatorBank() { free(oscillatorPhases); free(oscillatorNextNormFreq); free(oscillatorNextAmp); free(oscillatorNormFrequencies); free(oscillatorAmplitudes); free(oscillatorNormFreqDerivatives); free(oscillatorAmplitudeDerivatives); free(phaseCopies); free(nextNormFreqCopies); free(nextAmpCopies); delete[] oscStatNormFrequenciesMean; delete[] oscStatNumHops; delete[] lookupTable; delete[] indicesMapping; delete[] freqFixedDeltas; delete[] ampFixedDeltas; delete[] nyquistCut; } bool OscillatorBank::initBank(int oversamp) { if (!loaded) return false; //---prepare look-up table lookupTableSize = 1024; lookupTable = new float[lookupTableSize + 1]; for (int n = 0; n < (lookupTableSize + 1); n++) lookupTable[n] = sin(2.0 * M_PI * (float) n / (float) lookupTableSize); frequencyScaler = (float) lookupTableSize / rate; nyqNorm = rate / 2 * frequencyScaler; if (oversamp < 1) oversamp = 1; //---prepare oscillators partials = &(parser.partials); // pointer to paser's partials partialsHopSize = parser.getHopSize(); lastHop = partials->getHopNum(); // last bank hop is equal to last partial frame, which is equal to partial hop num overSampling = oversamp; hopSize = partialsHopSize / overSampling; // if oversampling, osc bank hop num > partials hop num hopSizeReminder = partialsHopSize % overSampling; oscBankHopSize = hopSize; numOfPartials = partials->getPartialNum(); numOfOscillators = partials->getMaxActivePartialNum(); // get the maximum number of active partials at the same time // set to next multiple of 4 [NEON] numOfOscillators = (numOfOscillators + 3) & ~0x3; // to be sure we can add up to 3 fake oscillators int err; //---allocate buffers // alligned buffers [NEON] err = posix_memalign((void**) &oscillatorPhases, 16, numOfOscillators * sizeof(float)); err += posix_memalign((void**) &oscillatorNextNormFreq, 16, numOfOscillators * sizeof(float)); err += posix_memalign((void**) &oscillatorNextAmp, 16, numOfOscillators * sizeof(float)); err += posix_memalign((void**) &oscillatorNormFrequencies, 16, numOfOscillators * sizeof(float)); err += posix_memalign((void**) &oscillatorAmplitudes, 16, numOfOscillators * sizeof(float)); err += posix_memalign((void**) &oscillatorNormFreqDerivatives, 16, numOfOscillators * sizeof(float)); err += posix_memalign((void**) &oscillatorAmplitudeDerivatives, 16, numOfOscillators * sizeof(float)); err += posix_memalign((void**) &phaseCopies, 16, numOfOscillators * sizeof(float)); err += posix_memalign((void**) &nextNormFreqCopies, 16, numOfOscillators * sizeof(float)); err += posix_memalign((void**) &nextAmpCopies, 16, numOfOscillators * sizeof(float)); // regular ones oscStatNormFrequenciesMean = new float[numOfPartials]; oscStatNumHops = new float[numOfPartials]; indicesMapping = new int[numOfPartials]; freqFixedDeltas = new float[numOfPartials]; ampFixedDeltas = new float[numOfPartials]; nyquistCut = new bool[numOfPartials]; if (err > 0) { dbox_printf("Failed memory allocations %@#!\n"); return false; } // copy stats [they do not change] for (int n = 0; n < numOfPartials; n++) { oscStatNormFrequenciesMean[n] = partials->partialFreqMean[n] * frequencyScaler; oscStatNumHops[n] = partials->partialNumFrames[n]; // in Parser and Partials "frames" are what we call here "hops" [see comment at top of file] } // deafult values actPartNum = 0; loopStartHop = 0; loopEndHop = (parser.partials.getHopNum() - 2) * overSampling; ampTh = 0.0001; hopNumTh = 0; pitchMultiplier = 1; freqMovement = 1; filterNum = 0; note = false; speed = 1; nextSpeed = -1; maxSpeed = 10; minSpeed = 0.1; jumpHop = -1; // filter filterMaxF = 22000; filterAmpMinF = 10 * frequencyScaler; filterAmpMaxF = 5000 * frequencyScaler; filterAmpMul = 10.0; // adsr minAttackTime = .0001; deltaAttackTime = 2.; minReleaseTime = 1; deltaReleaseTime = 2.5; adsr.setAttackRate(minAttackTime * rate); adsr.setDecayRate(.0001 * rate); adsr.setSustainLevel(1); adsr.setReleaseRate(minReleaseTime * rate); state = bank_stopped; return true; } void OscillatorBank::resetOscillators() { currentHop = -1; loopDir = 1; loopDirShift = 0; fill(nyquistCut, nyquistCut + numOfPartials, false); prevAdsrVal = 0; prevAmpTh = ampTh; prevHopNumTh = hopNumTh; prevPitchMultiplier = pitchMultiplier; prevFreqMovement = freqMovement; prevFilterNum = filterNum; memcpy(prevFilterFreqs, filterFreqs, filterNum * sizeof(float)); memcpy(prevFilterQ, filterQ, filterNum * sizeof(float)); int activePNum = partials->activePartialNum[0]; unsigned int *activeP = partials->activePartials[0]; for (int i = 0; i < activePNum; i++) { freqFixedDeltas[activeP[i]] = partials->partialFreqDelta[activeP[i]][0] / overSampling; ampFixedDeltas[activeP[i]] = partials->partialAmpDelta[activeP[i]][0] / overSampling; } // attack! adsr.gate(1); note = true; nextHop(); state = bank_playing; } void OscillatorBank::nextHop() { hopSize = oscBankHopSize; // copy phases, next freqs and next amps from previous frame memcpy(phaseCopies, oscillatorPhases, actPartNum * sizeof(float)); memcpy(nextNormFreqCopies, oscillatorNextNormFreq, actPartNum * sizeof(float)); memcpy(nextAmpCopies, oscillatorNextAmp, actPartNum * sizeof(float)); // next frame is forward or backwards, cos we could be in the loop currentHop += loopDir; checkDirection(); // if((currentHop/overSampling)%100 == 0) // dbox_printf("currentHop %d, direction: %d\n", currentHop/overSampling, loopDir); // if needs jump, end here this method, cos jumpToHop() will do tee rest if (checkJump() == 0) return; // otherwise, if jump is not needed or fails, continue regular stuff if (nextEnvState() != 0) return; // release has ended! checkSpeed(); // now let's decide how to calculate next hop if (!checkOversampling()) nextOscBankHop(); else nextPartialHop(); } void OscillatorBank::nextOscBankHop() { int parIndex, localHop; float parDamp = 1; int currentPartialHop = (currentHop / overSampling) + loopDirShift; // if going backwards in the loop, get previous frame active partials... actPartNum = partials->activePartialNum[currentPartialHop - loopDirShift]; actPart = partials->activePartials[currentPartialHop - loopDirShift]; //cout << "actPartNum: " << actPartNum << endl; envState = adsr.getState(); // to determine what state we will be in next hop [attack, decay, sustain, release] int parCnt = 0; int currentHopReminder = currentHop % overSampling; // steps to reach next bank hop from previous partial hop int steps = currentHopReminder + 1; if (loopDir < 0) steps = overSampling - currentHopReminder + 1; for (int i = 0; i < actPartNum; i++) { // find partial and frame parIndex = actPart[i]; //localHop = partials->localPartialFrames[currentPartialHop][parIndex]; localHop = currentPartialHop - partials->partialStartFrame[parIndex]; // in Parser and Partials "frames" are what we call here "hops". These particular ones are local frames [see comment at top of file] //float delta = partials->partialFrequencies[parIndex][localHop+loopDir] - partials->partialFrequencies[parIndex][localHop]; // if this partial was over nyquist on previous hop... if (nyquistCut[parIndex]) { // ...restart from safe values oscillatorPhases[parCnt] = 0; //TODO add freqmove dependency oscillatorNextNormFreq[parCnt] = (partials->partialFrequencies[parIndex][localHop] + freqFixedDeltas[parIndex] * (steps - 1)) * frequencyScaler * prevPitchMultiplier; oscillatorNextAmp[parCnt] = 0; } else if (loopDir == 1) // otherwise recover phase, target freq and target amp from previous frame { if ((localHop != 0) || (currentHopReminder != 0)) { oscillatorPhases[parCnt] = phaseCopies[indicesMapping[parIndex]]; oscillatorNextNormFreq[parCnt] = nextNormFreqCopies[indicesMapping[parIndex]]; oscillatorNextAmp[parCnt] = nextAmpCopies[indicesMapping[parIndex]]; } else // first oscillator hop [both for bank and partial], so no previous data are available { oscillatorPhases[parCnt] = 0; //TODO add freqmove dependency oscillatorNextNormFreq[parCnt] = partials->partialFrequencies[parIndex][localHop] * frequencyScaler * prevPitchMultiplier; parDamp = calculateParDamping(parIndex, prevHopNumTh, prevAdsrVal, oscillatorNextNormFreq[parCnt], prevFilterNum, prevFilterFreqs, prevFilterQ); oscillatorNextAmp[parCnt] = partials->partialAmplitudes[parIndex][localHop] * parDamp; if(oscillatorNextAmp[parCnt] > 1) oscillatorNextAmp[parCnt] = 1; freqFixedDeltas[parIndex] = partials->partialFreqDelta[parIndex][localHop + loopDir] * loopDir / overSampling; ampFixedDeltas[parIndex] = partials->partialAmpDelta[parIndex][localHop + loopDir] * loopDir / overSampling; } } else { oscillatorPhases[parCnt] = phaseCopies[indicesMapping[parIndex]]; oscillatorNextNormFreq[parCnt] = nextNormFreqCopies[indicesMapping[parIndex]]; oscillatorNextAmp[parCnt] = nextAmpCopies[indicesMapping[parIndex]]; } // remove aliasing, skipping partial over nyquist freq if (oscillatorNextNormFreq[parCnt] > nyqNorm) { nyquistCut[parIndex] = true; continue; } nyquistCut[parIndex] = false; // first set up freq, cos filter affects amplitude damping according to freq content oscillatorNormFrequencies[parCnt] = oscillatorNextNormFreq[parCnt]; // to fix any possible drifts // save next values, current for next round oscillatorNextNormFreq[parCnt] = (freqMovement * (partials->partialFrequencies[parIndex][localHop] + freqFixedDeltas[parIndex] * steps) * frequencyScaler + (1 - freqMovement) * oscStatNormFrequenciesMean[parIndex]) * pitchMultiplier; // derivatives are (next hop value*next damping) - (current hop value*current damping) ---> next hop must be available, in both directions, because of control on active partials oscillatorNormFreqDerivatives[parCnt] = (oscillatorNextNormFreq[parCnt] - oscillatorNormFrequencies[parCnt]) / hopCounter; // this second weird passage handles dissonance control, morphing between regular and mean frequencies oscillatorNormFreqDerivatives[parCnt] = freqMovement * oscillatorNormFreqDerivatives[parCnt] + (1 - freqMovement) * ((oscStatNormFrequenciesMean[parIndex] * pitchMultiplier) - oscillatorNormFrequencies[parCnt]) / hopCounter; parDamp = calculateParDamping(parIndex, hopNumTh, adsrVal, oscillatorNextNormFreq[parCnt], filterNum, filterFreqs, filterQ); // now amplitudes oscillatorAmplitudes[parCnt] = oscillatorNextAmp[parCnt]; // to fix any possible drifts // save next values, current for next round //delta = partials->partialAmplitudes[parIndex][localHop+loopDir] - partials->partialAmplitudes[parIndex][localHop]; oscillatorNextAmp[parCnt] = (partials->partialAmplitudes[parIndex][localHop] + ampFixedDeltas[parIndex] * steps) * parDamp; if(oscillatorNextAmp[parCnt] > 1) oscillatorNextAmp[parCnt] = 1; if ((loopDir == -1) && (localHop = 1) && (currentHopReminder == 1)) oscillatorNextAmp[parCnt] = 0; // derivatives are (next hop value*next damping) - (current hop value*current damping) ---> next hop must be available, in both directions, because of control on active partials oscillatorAmplitudeDerivatives[parCnt] = (oscillatorNextAmp[parCnt] - oscillatorAmplitudes[parCnt]) / hopCounter; // finally update current mapping between oscillators and partials indicesMapping[parIndex] = parCnt; parCnt++; } actPartNum = parCnt; // [NEON] if not multiple of 4... if (actPartNum % 4 != 0) addFakeOsc(); } void OscillatorBank::nextPartialHop() { unsigned int parIndex, localHop; float parDamp = 1; int currentPartialHop = currentHop / overSampling; // if going backwards in the loop, get previous frame active partials... actPartNum = partials->activePartialNum[currentPartialHop - loopDirShift]; actPart = partials->activePartials[currentPartialHop - loopDirShift]; envState = adsr.getState(); // to determine what state we will be in next hop [attack, decay, sustain, release] int parCnt = 0; int steps = overSampling - 1; // steps to reach next hop [partial or bank] from previous partial hop for (int i = 0; i < actPartNum; i++) { // find partial and frame parIndex = actPart[i]; //localHop = partials->localPartialFrames[currentPartialHop][parIndex]; localHop = currentPartialHop - partials->partialStartFrame[parIndex]; // in Parser and Partials "frames" are what we call here "hops". These particular ones are local frames [see comment at top of file] // if this partial was over nyquist on previous hop... if (nyquistCut[parIndex]) { // ...restart from safe values oscillatorPhases[parCnt] = 0; //TODO add freqmove dependency oscillatorNextNormFreq[parCnt] = (partials->partialFrequencies[parIndex][localHop] + freqFixedDeltas[parIndex] * steps * (1 - loopDirShift)) * frequencyScaler * prevPitchMultiplier; oscillatorNextAmp[parCnt] = 0; } else if (loopDir == 1) // otherwise recover phase, target freq and target amp from previous frame { if ((localHop != 0) || (overSampling > 1)) { oscillatorPhases[parCnt] = phaseCopies[indicesMapping[parIndex]]; oscillatorNextNormFreq[parCnt] = nextNormFreqCopies[indicesMapping[parIndex]]; oscillatorNextAmp[parCnt] = nextAmpCopies[indicesMapping[parIndex]]; } else // first oscillator hop [both for bank and partial], so no previous data are available { oscillatorPhases[parCnt] = 0; //TODO add freqmove dependency oscillatorNextNormFreq[parCnt] = partials->partialFrequencies[parIndex][localHop] * frequencyScaler * prevPitchMultiplier; parDamp = calculateParDamping(parIndex, prevHopNumTh, prevAdsrVal, oscillatorNextNormFreq[parCnt], prevFilterNum, prevFilterFreqs, prevFilterQ); oscillatorNextAmp[parCnt] = partials->partialAmplitudes[parIndex][localHop] * parDamp; if(oscillatorNextAmp[parCnt] > 1) oscillatorNextAmp[parCnt] = 1; freqFixedDeltas[parIndex] = partials->partialFreqDelta[parIndex][localHop + loopDir] * loopDir / overSampling; ampFixedDeltas[parIndex] = partials->partialAmpDelta[parIndex][localHop + loopDir] * loopDir / overSampling; } } else { if (localHop != partials->partialNumFrames[parIndex] - 1) { oscillatorPhases[parCnt] = phaseCopies[indicesMapping[parIndex]]; oscillatorNextNormFreq[parCnt] = nextNormFreqCopies[indicesMapping[parIndex]]; oscillatorNextAmp[parCnt] = nextAmpCopies[indicesMapping[parIndex]]; } else // first oscillator hop [going backwards - both for bank and partial] , so no previous data are available { oscillatorPhases[parCnt] = 0; //TODO add freqmove dependency oscillatorNextNormFreq[parCnt] = partials->partialFrequencies[parIndex][localHop] * frequencyScaler * prevPitchMultiplier; parDamp = calculateParDamping(parIndex, prevHopNumTh, prevAdsrVal, oscillatorNextNormFreq[parCnt], prevFilterNum, prevFilterFreqs, prevFilterQ); oscillatorNextAmp[parCnt] = partials->partialAmplitudes[parIndex][localHop] * parDamp; if(oscillatorNextAmp[parCnt] > 1) oscillatorNextAmp[parCnt] = 1; freqFixedDeltas[parIndex] = partials->partialFreqDelta[parIndex][localHop + loopDir] * loopDir / overSampling; ampFixedDeltas[parIndex] = partials->partialAmpDelta[parIndex][localHop + loopDir] * loopDir / overSampling; } } // remove aliasing, skipping partial over nyquist freq if (oscillatorNextNormFreq[parCnt] > nyqNorm) { //cout << nyqNorm << endl; nyquistCut[parIndex] = true; continue; } nyquistCut[parIndex] = false; // first set up freq, cos filter affects amplitude damping according to freq content oscillatorNormFrequencies[parCnt] = oscillatorNextNormFreq[parCnt]; // to fix any possible drifts // save next values, current for next round oscillatorNextNormFreq[parCnt] = (freqMovement * (partials->partialFrequencies[parIndex][localHop + loopDir] - freqFixedDeltas[parIndex] * steps * loopDirShift) * frequencyScaler + (1 - freqMovement) * oscStatNormFrequenciesMean[parIndex]) * pitchMultiplier; // derivatives are (next hop value*next damping) - (current hop value*current damping) ---> next hop must be available, in both directions, because of control on active partials oscillatorNormFreqDerivatives[parCnt] = (oscillatorNextNormFreq[parCnt] - oscillatorNormFrequencies[parCnt]) / hopCounter; // this second weird passage handles dissonance control, morphing between regular and mean frequencies oscillatorNormFreqDerivatives[parCnt] = freqMovement * oscillatorNormFreqDerivatives[parCnt] + (1 - freqMovement) * ((oscStatNormFrequenciesMean[parIndex] * pitchMultiplier) - oscillatorNormFrequencies[parCnt]) / hopCounter; parDamp = calculateParDamping(parIndex, hopNumTh, adsrVal, oscillatorNextNormFreq[parCnt], filterNum, filterFreqs, filterQ); // now amplitudes oscillatorAmplitudes[parCnt] = oscillatorNextAmp[parCnt]; // to fix any possible drifts // save next values, current for next round //delta = partials->partialAmplitudes[parIndex][localHop+loopDir] - partials->partialAmplitudes[parIndex][localHop]; oscillatorNextAmp[parCnt] = (partials->partialAmplitudes[parIndex][localHop + loopDir] - (ampFixedDeltas[parIndex]) * steps * loopDirShift) * parDamp; if(oscillatorNextAmp[parCnt] > 1) oscillatorNextAmp[parCnt] = 1; // to avoid bursts when transients are played backwards if ((loopDir == -1) && (localHop - 1 == 0) && (overSampling == 1)) { oscillatorNextAmp[parCnt] = 0; } // derivatives are (next hop value*next damping) - (current hop value*current damping) ---> next hop must be available, in both directions, because of control on active partials oscillatorAmplitudeDerivatives[parCnt] = (oscillatorNextAmp[parCnt] - oscillatorAmplitudes[parCnt]) / hopCounter; // if next is not going to loop boundaries, get next deltas [same direction] if ((((currentPartialHop + loopDir) * overSampling != loopEndHop) || (loopDir == -1)) && (((currentPartialHop + loopDir) * overSampling + loopDir != loopStartHop) || (loopDir == 1))) { freqFixedDeltas[parIndex] = partials->partialFreqDelta[parIndex][localHop + loopDir] * loopDir / overSampling; ampFixedDeltas[parIndex] = partials->partialAmpDelta[parIndex][localHop + loopDir] * loopDir / overSampling; } else // .. otherwise, keep deltas but change sign [co swe change direction] { freqFixedDeltas[parIndex] = -freqFixedDeltas[parIndex]; ampFixedDeltas[parIndex] = -ampFixedDeltas[parIndex]; } // finally update current mapping between oscillators and partials indicesMapping[parIndex] = parCnt; parCnt++; } actPartNum = parCnt; // [NEON] if not multiple of 4... if (actPartNum % 4 != 0) addFakeOsc(); updatePrevControls(); } void OscillatorBank::addFakeOsc() { // ...calculate difference int newPartNum = (actPartNum + 3) & ~0x3; // ...add fake oscillators until total num is multiple of 4 for (int i = actPartNum; i < newPartNum; i++) { oscillatorAmplitudes[i] = 0; oscillatorNormFrequencies[i] = 0; oscillatorAmplitudeDerivatives[i] = 0; oscillatorNormFreqDerivatives[i] = 0; oscillatorPhases[i] = 0; } // ...and update num of active partials actPartNum = newPartNum; } void OscillatorBank::play(float vel) { // set attack and release params according to velocity //adsr.setAttackRate((minAttackTime + ((1 - vel) * deltaAttackTime)) * rate); adsr.setAttackRate(minAttackTime * rate); //adsr.setReleaseRate((minReleaseTime + (1 - vel) * deltaReleaseTime) * rate); adsr.setReleaseRate(minReleaseTime * rate); // set timbre hopNumTh = log((1 - vel) + 1) / log(2) * 20000; state = bank_toreset; } //--------------------------------------------------------------------------------------------------------------------------- // private methods //--------------------------------------------------------------------------------------------------------------------------- bool OscillatorBank::loader(char *filename, int hopsize, int samplerate) { rate = samplerate; loaded = parser.parseFile(filename, hopsize, samplerate); return loaded; } int OscillatorBank::jumpToHop() { int jumpGap = abs(jumpHop - currentHop / overSampling); // gaps in partial reference // can't jump to self dude if (jumpGap == 0) return 1; // direction is in general maintained with jump if (jumpHop == 0) setDirection(1); else if (jumpHop == lastHop) setDirection(-1); dbox_printf("\tJump from %d to %d\n", currentHop / overSampling, jumpHop); dbox_printf("\tdirection %d\n", loopDir); currentHop = jumpHop * overSampling; if (nextEnvState() != 0) return 0; // release has ended! checkSpeed(); int parIndex, localHop, targetHop; float parDamp = 1; int currentPartialHop = currentHop / overSampling; int targetPartialHop = jumpHop; actPartNum = partials->activePartialNum[currentPartialHop]; actPart = partials->activePartials[currentPartialHop]; int targetActParNum = partials->activePartialNum[targetPartialHop]; unsigned int *targetActPar = partials->activePartials[targetPartialHop]; envState = adsr.getState(); // to determine what state we will be in next hop [attack, decay, sustain, release] int parCnt = 0; int currentHopReminder = currentHop % overSampling; // steps to walk where i am [bank of partial hop] from previous partial hop int steps = currentHopReminder * (overSampling != 1); // no oversampling 0, oversampling and going ff currentHopReminder for (int i = 0; i < actPartNum; i++) { // find partial and frame parIndex = actPart[i]; //localHop = partials->localPartialFrames[currentPartialHop][parIndex]; localHop = currentPartialHop - partials->partialStartFrame[parIndex]; // in Parser and Partials "frames" are what we call here "hops". These particular ones are local frames [see comment at top of file] // if this partial was over nyquist on previous hop... if (nyquistCut[parIndex]) { // ...restart from safe values oscillatorPhases[parCnt] = 0; //TODO add freqmove dependency oscillatorNextNormFreq[parCnt] = (partials->partialFrequencies[parIndex][localHop] + freqFixedDeltas[parIndex] * steps * loopDir) * frequencyScaler * prevPitchMultiplier; oscillatorNextAmp[parCnt] = 0; } else if (loopDir == 1) {// otherwise recover phase, target freq and target amp from previous frame if ((localHop != 0) || ((overSampling > 1) && (currentHopReminder != 0))) { oscillatorPhases[parCnt] = phaseCopies[indicesMapping[parIndex]]; oscillatorNextNormFreq[parCnt] = nextNormFreqCopies[indicesMapping[parIndex]]; oscillatorNextAmp[parCnt] = nextAmpCopies[indicesMapping[parIndex]]; } else { // first oscillator hop [both for bank and partial], so no previous data are available oscillatorPhases[parCnt] = 0; //TODO add freqmove dependency oscillatorNextNormFreq[parCnt] = partials->partialFrequencies[parIndex][localHop] * frequencyScaler * prevPitchMultiplier; parDamp = calculateParDamping(parIndex, prevHopNumTh, prevAdsrVal, oscillatorNextNormFreq[parCnt], prevFilterNum, prevFilterFreqs, prevFilterQ); oscillatorNextAmp[parCnt] = partials->partialAmplitudes[parIndex][localHop] * parDamp; if(oscillatorNextAmp[parCnt] > 1) oscillatorNextAmp[parCnt] = 1; } } else { if (( (unsigned)localHop != partials->partialNumFrames[parIndex] - 1) || ((overSampling > 1) && (currentHopReminder != 0))) { oscillatorPhases[parCnt] = phaseCopies[indicesMapping[parIndex]]; oscillatorNextNormFreq[parCnt] = nextNormFreqCopies[indicesMapping[parIndex]]; oscillatorNextAmp[parCnt] = nextAmpCopies[indicesMapping[parIndex]]; } else // first oscillator hop [going backwards - both for bank and partial] , so no previous data are available, so retrieve where i am { oscillatorPhases[parCnt] = 0; //TODO add freqmove dependency oscillatorNextNormFreq[parCnt] = partials->partialFrequencies[parIndex][localHop] * frequencyScaler * prevPitchMultiplier; parDamp = calculateParDamping(parIndex, prevHopNumTh, prevAdsrVal, oscillatorNextNormFreq[parCnt], prevFilterNum, prevFilterFreqs, prevFilterQ); oscillatorNextAmp[parCnt] = partials->partialAmplitudes[parIndex][localHop] * parDamp; if(oscillatorNextAmp[parCnt] > 1) oscillatorNextAmp[parCnt] = 1; } } // remove aliasing, skipping partial over nyquist freq if (oscillatorNextNormFreq[parCnt] > nyqNorm) { //cout << nyqNorm << endl; nyquistCut[parIndex] = true; continue; } nyquistCut[parIndex] = false; // check what happens of this partial at target hop float targetFreqVal, targetAmpVal; //targetHop = partials->localPartialFrames[targetPartialHop][parIndex]; targetHop = targetPartialHop - partials->partialStartFrame[parIndex]; if (targetHop == -1) targetFreqVal = targetAmpVal = 0; else { targetFreqVal = partials->partialFrequencies[parIndex][targetHop] * frequencyScaler; // pitch shift will be multiplied later!!! targetAmpVal = partials->partialFrequencies[parIndex][targetHop]; // parDamp will be multiplied later!!! } // first set up freq, cos filter affects amplitude damping according to freq content oscillatorNormFrequencies[parCnt] = oscillatorNextNormFreq[parCnt]; // to fix any possible drifts // save next values, current for next round oscillatorNextNormFreq[parCnt] = (freqMovement * targetFreqVal + (1 - freqMovement) * oscStatNormFrequenciesMean[parIndex]) * pitchMultiplier; // derivatives are (next hop value*next damping) - (current hop value*current damping) ---> next hop must be available, in both directions, because of control on active partials oscillatorNormFreqDerivatives[parCnt] = (oscillatorNextNormFreq[parCnt] - oscillatorNormFrequencies[parCnt]) / hopCounter; // this second weird passage handles dissonance control, morphing between regular and mean frequencies oscillatorNormFreqDerivatives[parCnt] = freqMovement * oscillatorNormFreqDerivatives[parCnt] + (1 - freqMovement) * ((oscStatNormFrequenciesMean[parIndex] * pitchMultiplier) - oscillatorNormFrequencies[parCnt]) / hopCounter; parDamp = calculateParDamping(parIndex, hopNumTh, adsrVal, oscillatorNextNormFreq[parCnt], filterNum, filterFreqs, filterQ); // now amplitudes oscillatorAmplitudes[parCnt] = oscillatorNextAmp[parCnt]; // to fix any possible drifts // save next values, current for next round oscillatorNextAmp[parCnt] = targetAmpVal * parDamp; if(oscillatorNextAmp[parCnt] > 1) oscillatorNextAmp[parCnt] = 1; // to avoid bursts when transients are played backwards if ((loopDir == -1) && (targetHop == 0) && ((overSampling == 1) || (currentHopReminder == 0))) { oscillatorNextAmp[parCnt] = 0; } // derivatives are (next hop value*next damping) - (current hop value*current damping) ---> next hop must be available, in both directions, because of control on active partials oscillatorAmplitudeDerivatives[parCnt] = (oscillatorNextAmp[parCnt] - oscillatorAmplitudes[parCnt]) / hopCounter; //if partial does not die at target, calculate deltas according to direction if (targetHop != -1) { freqFixedDeltas[parIndex] = partials->partialFreqDelta[parIndex][targetHop] * loopDir / overSampling; ampFixedDeltas[parIndex] = partials->partialAmpDelta[parIndex][targetHop] * loopDir / overSampling; } // finally update current mapping between oscillators and partials indicesMapping[parIndex] = parCnt; parCnt++; } actPartNum = parCnt; // now add the ones that start at target hop! for (int i = 0; i < targetActParNum; i++) { // find partial and frame parIndex = targetActPar[i]; //targetHop = partials->localPartialFrames[targetPartialHop][parIndex]; targetHop = targetPartialHop - partials->partialStartFrame[parIndex]; // in Parser and Partials "frames" are what we call here "hops". These particular ones are local frames [see comment at top of file] // check if this partials was already active before the jump //localHop = partials->localPartialFrames[currentPartialHop][parIndex]; localHop = currentPartialHop - partials->partialStartFrame[parIndex]; // if yes, skip it if (localHop != -1) continue; // otherwise add it to active bunch and calcucalte values // first set up freq, cos filter affects amplitude damping according to freq content oscillatorNormFrequencies[parCnt] = 0; // save next values, current for next round oscillatorNextNormFreq[parCnt] = (freqMovement * partials->partialFrequencies[parIndex][targetHop] * frequencyScaler + (1 - freqMovement) * oscStatNormFrequenciesMean[parIndex]) * pitchMultiplier; // derivatives are (next hop value*next damping) - (current hop value*current damping) ---> next hop must be available, in both directions, because of control on active partials oscillatorNormFreqDerivatives[parCnt] = (oscillatorNextNormFreq[parCnt] - oscillatorNormFrequencies[parCnt]) / hopCounter; // this second weird passage handles dissonance control, morphing between regular and mean frequencies oscillatorNormFreqDerivatives[parCnt] = freqMovement * oscillatorNormFreqDerivatives[parCnt] + (1 - freqMovement) * ((oscStatNormFrequenciesMean[parIndex] * pitchMultiplier) - oscillatorNormFrequencies[parCnt]) / hopCounter; parDamp = calculateParDamping(parIndex, hopNumTh, adsrVal, oscillatorNextNormFreq[parCnt], filterNum, filterFreqs, filterQ); // now amplitudes oscillatorAmplitudes[parCnt] = 0; // save next values, current for next round oscillatorNextAmp[parCnt] = partials->partialFrequencies[parIndex][targetHop] * parDamp; if(oscillatorNextAmp[parCnt] > 1) oscillatorNextAmp[parCnt] = 1; // derivatives are (next hop value*next damping) - (current hop value*current damping) ---> next hop must be available, in both directions, because of control on active partials oscillatorAmplitudeDerivatives[parCnt] = (oscillatorNextAmp[parCnt] - oscillatorAmplitudes[parCnt]) / hopCounter; //calculate deltas according to direction freqFixedDeltas[parIndex] = partials->partialFreqDelta[parIndex][targetHop] * loopDir / overSampling; ampFixedDeltas[parIndex] = partials->partialAmpDelta[parIndex][targetHop] * loopDir / overSampling; // finally update current mapping between oscillators and partials indicesMapping[parIndex] = parCnt; parCnt++; } // [NEON] if not multiple of 4... if (actPartNum % 4 != 0) addFakeOsc(); updatePrevControls(); jumpHop = -1; return 0; } int OscillatorBank::nextEnvState() { /* envState = Attack.getState(); // to determine what state we are in [attack, decay, sustain, release] // osc bank is playing the tail and the tail ends... if( (state == bank_playing)&&(envState == env_idle) ) { state = bank_stopped; // ...stop bank return 1; // and return immediately } else if( (envState == env_attack) || (envState == env_decay) ) { // run envelopes until next frame dampWeight = Attack.process(hopSize); } else if(envState == env_release) { // run envelopes until next frame dampWeight = Attack.process(hopSize); releaseDamp = Release.process(hopSize); }*/ envState = adsr.getState(); // osc bank is playing the tail and the tail ends... if ((state == bank_playing) && (envState == env_idle)) { state = bank_stopped; // ...stop bank adsrVal = 0; return 1; // and return immediately } else adsrVal = adsr.process(hopSize); return 0; } void OscillatorBank::checkDirection() { // end of the loop or end of file if (((currentHop >= loopEndHop) && (loopDir == 1)) || ((currentHop >= lastHop) && (loopDir == 1))) { // move backwards setDirection(-1); //dbox_printf("backward from %d\n", loopEndHop); } else if (((currentHop <= loopStartHop) && (loopDir == -1)) || ((currentHop <= 0) && (loopDir == -1))) // start of the loop or start of file { // move forward setDirection(1); //dbox_printf("forward from %d\n", loopStartHop); } } void OscillatorBank::checkSpeed() { // speed control [alike on highways, LOL] if (nextSpeed > 0) { nextSpeed = (nextSpeed < maxSpeed) ? nextSpeed : maxSpeed; nextSpeed = (nextSpeed > minSpeed) ? nextSpeed : minSpeed; speed = nextSpeed; nextSpeed = -1; } hopCounter = hopSize / speed; } int OscillatorBank::checkJump() { //check if has to jump somewhere if (jumpHop > -1) { // needs to jump! if (jumpToHop() == 0) return 0; } return 1; // no jump } bool OscillatorBank::checkOversampling() { //TODO fix this, but need andrew to fix oversampling multiple of period size // if partialsHopSize is not a multiple of oversampling, change hop size to periodically match next partial hop if (hopSizeReminder > 0) { // if next osc bank hop overtakes next partial hop... if ((currentHop + loopDir) * hopSize > partialsHopSize) { hopSize = hopSizeReminder; // ...shrink osc bank hop size to match partial hop return true; // and set next hop as matching with next partial hop } } else if (((currentHop + (1 - loopDirShift)) % overSampling) == 0) // if next osc bank hop matches next partial hop return true; // ...mark next hop as partial hop return false; // ,otherwise mark next hop as osc bank hop } void OscillatorBank::updatePrevControls() { prevAdsrVal = adsrVal; prevAmpTh = ampTh; prevHopNumTh = hopNumTh; prevPitchMultiplier = pitchMultiplier; prevFreqMovement = freqMovement; prevFilterNum = filterNum; memcpy(prevFilterFreqs, filterFreqs, filterNum * sizeof(float)); memcpy(prevFilterQ, filterQ, filterNum * sizeof(float)); } float OscillatorBank::calculateParDamping(int parIndex, int hopNTh, float adsrVl, float nextFreq, int filNum, float *filFreq, float *filQ) { float parDamp = 1; // timbre parDamp = ((float) (oscStatNumHops[parIndex] + 1)) / (hopNTh + 1); parDamp = (parDamp > 1) ? 1 : parDamp; parDamp = adsrVl * parDamp; //filters float filterWeights[MAX_TOUCHES]; float filterDamp[MAX_TOUCHES]; float filDist; float filterWeightsAcc; float filDmp; float filAmp; // band reject notch filter // float dist, dmp; // for(int k=0; k<filterNum; k++) // { // dist = fabs(oscillatorNextNormFreq[parCnt]-filterFreqs[k]); // if(dist<=filterQ[k]) // { // dmp = dist/filterQ[k]; // parDamp *= dmp*dmp*dmp; // } // } // each filter is a band pass notch filter // if at least one is active if (filNum > 0) { // reset values filDist = 0; filterWeightsAcc = 0; filDmp = 0; filAmp = 0; // for each filter for (int k = 0; k < filNum; k++) { // here are a couple of kludges to boost sound output of hi freq filters // damping effect of filter increases with distance, but decreases with filter frequency [kludge] float mul = ((filterMaxF-nextFreq)/filterMaxF) * 0.9 + 0.1 ; //filDist = fabs(nextFreq - filFreq[k])*( ((exp(a*4)-1)/EXP_DENOM) * 0.9 + 0.1 ); filDist = fabs(nextFreq - filFreq[k])*mul; // these to merge all filters contributions according to distance filterWeights[k] = filterMaxF - filDist; filterWeightsAcc += filterWeights[k]; // freqs very close to filter center are slightly amplified // the size of this amp area and the effect of amplification increase with frequency [kludge] if (filDist < filterAmpMinF + (filterAmpMaxF*(1-mul) - filterAmpMinF) * (1 - filQ[k]) ) filAmp = filQ[k] * filterAmpMul*(1-mul); else filAmp = 0; // actual damping filDmp = 1 / (filDist * filQ[k]); filDmp = (filDmp > 1) ? 1 : filDmp; // sum damp+amplification filterDamp[k] = filDmp + filAmp; } // do weighted mean to merge all filters contributions filDmp = 0; for (int k = 0; k < filNum; k++) filDmp += filterDamp[k] * filterWeights[k]; filDmp /= filterWeightsAcc; // apply parDamp *= filDmp; } return parDamp; }