Compare commits

..

No commits in common. "cpuboard" and "master" have entirely different histories.

13 changed files with 31 additions and 10031 deletions

View File

@ -10,7 +10,6 @@
NAME = chassis NAME = chassis
FILES_DSP = \ FILES_DSP = \
ic1.cpp \
parameters.cpp \ parameters.cpp \
voicecpu.cpp \ voicecpu.cpp \
chassis.cpp \ chassis.cpp \

File diff suppressed because it is too large Load Diff

View File

@ -1,69 +0,0 @@
10 CALF $0D45 // look up digit value
1 CALF $085C // send "Zero DAC" to module board
1 CALF $087E // save edited value and send to sysex and module
1 CALF $08E9 // send sysex header if manual pressed
1 CALF $0955 // send pitchbend to midi
1 CALF $097E // send modwheel to midi
1 CALF $0996 // to do with voice functions
1 CALF $0A18 // to do with voice functions
1 CALF $0A76 // to do with voice functions
1 CALF $0A88 // voice functions
1 CALF $0AAF // voice functions
1 CALF $0AF8 // voice functions
1 CALF $0B30 // voice functions
1 CALF $0BA9 // Sustain on/off
1 CALF $0BCC // range switch values
1 CALF $0BDA // osc switch values
1 CALF $0BEB // both range and osc values
1 CALF $0C0C // switches2 values
1 CALF $0C54 // switches values
1 CALF $0F27 // maybe tape?
2 CALF $0800 // display
2 CALF $0813 // set flags and store midi channel
2 CALF $0871 // read ADCs
2 CALF $08C3 // unknown1
2 CALF $0936 // calculate patch number for MIDI
2 CALF $0947 // send to module board
2 CALF $09C2 // send note on message to MIDI
2 CALF $09DC // add midi channel and send
2 CALF $09E6_setRunningStatus
2 CALF $0A0D // to do with note on flags
2 CALF $0B55 // unison note on to module board
2 CALF $0C47 // manual mode middle bars
2 CALF $0CA7 // calculate address for patch data
2 CALF $0CF1 // transmit 8 bytes to module board
2 CALF $0D06 // something to do with patches
2 CALF $0D25 // encode switches byte into upper bits
2 CALF $0EB4 // unknown2
2 CALF $0EC3 // unknown2
2 CALF $0EE7 // unknown3
2 CALF $0EFB // memory protect switch
2 CALF $0F64 // unknown4
3 CALF $0837 // send B, A to module board
3 CALF $0843_sendOmniOffPoly
3 CALF $08D6 // send single value sysex
3 CALF $0915_sendSysexHeader
3 CALF $0B9E // clear note on bits at FF30
3 CALF $0D33 // save digit pattern
3 CALF $0D3C // restore digit pattern
3 CALF $0ED5 // unknown5
4 CALF $0828 // centre bend to module board
4 CALF $08f0_sendSysexPatch
4 CALF $0929 // send program change over MIDI
4 CALF $0B81 // voice allocation
4 CALF $0B9A // clear MIDI on bits
4 CALF $0BF9 // send switches 1 to module board
4 CALF $0C29 // send switches 2 to module board
4 CALF $0C9D // shift bits in A and count
4 CALF $0F01 // detect two keys pressed together, maybe poly1+2
4 CALF $0F18 // unknown6, probably tape
5 CALF $09D5 // omni off running status
5 CALF $0C3C // something to do with testing switches
5 CALF $0F38 // unknown7 very likely tape
6 CALF $0B70_stopAllNotes
6 CALF $0C93 // decimal points off?
6 CALF $0CD0 // patch change
6 CALF $0F07 // possibly to do with tape
7 CALF $0D5B // decimal points on?
7 CALF $0EF0 // something to do with LEDs, possibly to do with tape?
85 CALF $09E8_sendToTx

View File

@ -18,11 +18,8 @@
#include "chassis.hpp" #include "chassis.hpp"
#include "ic1.hpp"
#include "patches.hpp" #include "patches.hpp"
Assigner ic1;
START_NAMESPACE_DISTRHO START_NAMESPACE_DISTRHO
Chassis::Chassis() : Plugin(parameterCount, 128, 0), sampleRate(getSampleRate()) { // one parameter, no programs, no states Chassis::Chassis() : Plugin(parameterCount, 128, 0), sampleRate(getSampleRate()) { // one parameter, no programs, no states
@ -55,16 +52,6 @@ void Chassis::activate() {
for (uint8_t i = 0; i < 104; i++) { for (uint8_t i = 0; i < 104; i++) {
s.pitchCV[i] = (440.0f * powf(2, (i - 49) / 12.0f)) / sampleRate; s.pitchCV[i] = (440.0f * powf(2, (i - 49) / 12.0f)) / sampleRate;
} }
// resetvoices
for (uint8_t i=0; i<8; i++) {
s.noteTbl[i] = 0x80;
s.voiceTbl[i] = i;
s.voice[i].phase = (0.7*((rand()%65536)/65536.0f));//(60*s.voice[i].omega*i);
printf("%f %d\n", s.voice[i].phase, i);
}
} }
void Chassis::deactivate() { void Chassis::deactivate() {
@ -73,33 +60,33 @@ void Chassis::deactivate() {
} }
void Chassis::noteOn(uint8_t note) { void Chassis::noteOn(uint8_t note) {
uint32_t i;
// needed for LFO for now
s.keyon = true; s.keyon = true;
for (i = 0; i < NUM_VOICES; i++) {
// Unison Voice On vPtr++;
for (uint8_t i=0; i<8; i++ ) { if (vPtr == NUM_VOICES) vPtr = 0;
//s.noteTbl[i] = note; if (s.voice[vPtr].isFree()) {
s.voice[i].on(note, false); // if it's an existing note don't reset
s.voice[i].voicenum=i; s.voice[vPtr].on(note, s.voice[i].note != note);
// printf("%f %i\n", s.voice[i].phase, i); break;
}
} }
//s.voice[vPtr].on(note, true); if (i == NUM_VOICES) { // didn't find a free voice
vPtr++;
if (vPtr == NUM_VOICES) vPtr = 0;
s.voice[vPtr].on(note, true);
}
} }
void Chassis::noteOff(uint8_t note) { void Chassis::noteOff(uint8_t note) {
s.keyon = false;
// Unison Voice On for (uint32_t i = 0; i < NUM_VOICES; i++) {
for (uint8_t i=0; i<8; i++ ) { if (s.voice[i].note == note && !s.voice[i].isFree()) {
//s.noteTbl[i] = note;
s.voice[i].off(); s.voice[i].off();
break;
}
} }
// s.voice[i].off();
s.keyon=false;
} }
void Chassis::doMidi(const MidiEvent *ev, uint32_t count, uint32_t timeLimit) { void Chassis::doMidi(const MidiEvent *ev, uint32_t count, uint32_t timeLimit) {
@ -117,8 +104,6 @@ void Chassis::doMidi(const MidiEvent *ev, uint32_t count, uint32_t timeLimit) {
} }
} }
lastEvent = i; lastEvent = i;
//ic1.voiceOn();
} }
void Chassis::run(const float **, float **outputs, uint32_t frames, const MidiEvent *midiEvents, uint32_t midiEventCount) { void Chassis::run(const float **, float **outputs, uint32_t frames, const MidiEvent *midiEvents, uint32_t midiEventCount) {
@ -174,26 +159,6 @@ void Chassis::run(const float **, float **outputs, uint32_t frames, const MidiEv
s.blockLeft -= sizeThisTime; s.blockLeft -= sizeThisTime;
} }
// highpass filter
float tmp = s.hptmp;
float smp, flt = 0;
float cut_tbl[4] = {0.91245, 0.97097, 1, 0.99088};
float cut = cut_tbl[(s.patchRam.switch2 & 0x18) >> 3];
float gain_tbl[4] = {-1, -1, 0, 4};
float gain = gain_tbl[(s.patchRam.switch2 & 0x18) >> 3];
// printf("hp= %02x\n", (s.patchRam.switch2 & 0x18)>>3);
for (uint32_t i = 0; i < frames; i++) {
smp = outputs[0][i];
flt = ((tmp - smp) * cut) + smp;
tmp = flt;
outputs[0][i] = ((flt * gain) + smp);
}
s.hptmp = flt;
// copy left to right // copy left to right
memmove(outputs[1], outputs[0], sizeof(float) * frames); memmove(outputs[1], outputs[0], sizeof(float) * frames);
// outputs[1][1]=1; // outputs[1][1]=1;

View File

@ -61,14 +61,8 @@ class Chassis : public Plugin {
paramSustain, paramSustain,
paramRelease, paramRelease,
paramChorusMode,
paramPolyMode,
paramUnisonDetune,
paramModWheel, paramModWheel,
parameterCount parameterCount
}; };

File diff suppressed because it is too large Load Diff

View File

@ -1,48 +0,0 @@
/*
Chassis polysynth framework
Copyright 2024 Gordon JC Pearce <gordonjcp@gjcp.net>
Permission to use, copy, modify, and/or distribute this software for any
purpose with or without fee is hereby granted, provided that the above
copyright notice and this permission notice appear in all copies.
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION
OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
*/
#pragma once
#include <cstdint>
class Assigner {
public:
void noteOn(uint8_t c);
private:
// 256 bytes of RAM from $ff00 to $ffff
uint8_t ram[256];
// bitmask data is at $0008 and $0010
// eight padding bytes at the start
uint8_t bitMask[24] = {
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
0x01, 0x02, 0x04, 0x08, 0x10, 0x20, 0x40, 0x80,
0xfe, 0xfd, 0xfb, 0xf7, 0xef, 0xdf, 0xbf, 0x7f};
uint16_t a = 0, b = 0, c = 0, eal = 0, eah = 0;
uint32_t bc, de, hl, ea = 0;
/*
23:31 < Alipha> gordonjcp: struct reg16 { uint16_t value; uint8_t hi() const { return value >> 8; } uint8_t lo() const { return value;
} };
23:31 < gordonjcp> Alipha
*/
};

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

View File

@ -149,7 +149,7 @@ void Chassis::initParameter(uint32_t index, Parameter& parameter) {
parameter.name = "HPF"; parameter.name = "HPF";
parameter.symbol = "ch_hpf"; parameter.symbol = "ch_hpf";
parameter.ranges.min = 0.0f; parameter.ranges.min = 0.0f;
parameter.ranges.max = 3.9f; parameter.ranges.max = 3.0f;
parameter.ranges.def = 0.0f; parameter.ranges.def = 0.0f;
parameter.midiCC = 20; parameter.midiCC = 20;
break; break;
@ -290,50 +290,6 @@ void Chassis::initParameter(uint32_t index, Parameter& parameter) {
parameter.midiCC = 26; parameter.midiCC = 26;
break; break;
case paramChorusMode:
parameter.hints = kParameterIsAutomatable | kParameterIsInteger;
parameter.name = "Chorus";
parameter.symbol = "ch_chorusmode";
parameter.ranges.min = 0.0f;
parameter.ranges.max = 2.0f;
parameter.ranges.def = 1.0f;
parameter.midiCC = 93;
break;
/*
case paramPolyMode:
parameter.hints = kParameterIsAutomatable | kParameterIsInteger;
parameter.name = "Poly";
parameter.symbol = "ch_poly";
parameter.ranges.min = 0.0f;
parameter.ranges.max = 2.0f;
parameter.ranges.def = 0.0f;
parameter.midiCC = 86;
parameter.enumValues.count = 3;
parameter.enumValues.restrictedMode = true;
{
ParameterEnumerationValue* const enumValues = new ParameterEnumerationValue[3];
enumValues[0].value = 0.0f;
enumValues[0].label = "16'";
enumValues[1].value = 1.0f;
enumValues[1].label = "8'";
enumValues[2].value = 2.0f;
enumValues[2].label = "4'";
parameter.enumValues.values = enumValues;
}
break;
*/
case paramUnisonDetune:
parameter.hints = kParameterIsAutomatable;
parameter.name = "Detune";
parameter.symbol = "ch_detune";
parameter.ranges.min = 0.0f;
parameter.ranges.max = 1.0f;
parameter.ranges.def = 0.2f;
parameter.midiCC=84;
break;
case paramModWheel: case paramModWheel:
parameter.hints = kParameterIsAutomatable | kParameterIsHidden; parameter.hints = kParameterIsAutomatable | kParameterIsHidden;
parameter.name = "Mod wheel"; parameter.name = "Mod wheel";
@ -341,7 +297,7 @@ void Chassis::initParameter(uint32_t index, Parameter& parameter) {
parameter.ranges.min = 0.0f; parameter.ranges.min = 0.0f;
parameter.ranges.max = 127.0f; parameter.ranges.max = 127.0f;
parameter.ranges.def = 0.0f; parameter.ranges.def = 0.0f;
//parameter.midiCC = 1; parameter.midiCC = 1;
break; break;
} }
// chorus, porta, bend range, key mode still to do // chorus, porta, bend range, key mode still to do
@ -441,22 +397,14 @@ void Chassis::setParameterValue(uint32_t index, float value) {
case paramHPF: // bits 3-4 of switch 2 case paramHPF: // bits 3-4 of switch 2
// doesn't look great in Carla because of odd behaviour with small integer knobs // doesn't look great in Carla because of odd behaviour with small integer knobs
printf("setPV %d %f\n", index, value);
if (value > 3) value = 3; if (value > 3) value = 3;
s.patchRam.switch2 &= 0xe7; s.patchRam.switch2 &= 0xf3;
s.patchRam.switch2 |= (3 - (int)value) << 3; s.patchRam.switch2 |= (int)value << 3;
break; break;
case paramModWheel: case paramModWheel:
s.ff64 = (int)value << 1; s.ff64 = (int)value << 1;
break; break;
//case paramPolyMode:
// s.polymode = (int)value;
// break;
case paramUnisonDetune:
s.unisonDetune = value;
break;
} }
} }
@ -504,7 +452,7 @@ float Chassis::getParameterValue(uint32_t index) const {
return s.patchRam.noise; return s.patchRam.noise;
break; break;
case paramHPF: case paramHPF:
return 3 - ((s.patchRam.switch2 & 0x18) >> 3); return (s.patchRam.switch2 & 0x18) >> 3;
break; break;
case paramVCFFreq: case paramVCFFreq:
return s.patchRam.vcfFreq; return s.patchRam.vcfFreq;
@ -543,10 +491,6 @@ float Chassis::getParameterValue(uint32_t index) const {
case paramVCALevel: case paramVCALevel:
return s.patchRam.vca; return s.patchRam.vca;
break;
case paramUnisonDetune:
return s.unisonDetune;
break; break;
} }
return 0; return 0;

View File

@ -42,21 +42,17 @@ void Voice::run(Synth &s, float *buffer, uint32_t samples) {
float fb, res = s.patchRam.vcfReso / 28.0f; // guess float fb, res = s.patchRam.vcfReso / 28.0f; // guess
float cut = 248.0f * (powf(2, (vcfenv - 0x1880) / 1143.0f)); float cut = 248.0f * (powf(2, (vcfenv - 0x1880) / 1143.0f));
cut *= (1-detune[voicenum]);
// now radians // now radians
cut = 0.25 * 6.2832 * cut / 48000.0f; cut = 0.25 * 6.2832 * cut / 48000.0f;
// now correct // now correct
cut = cut / (1 + cut); cut = cut / (1 + cut);
float sqr = (s.patchRam.switch1 & 0x08) ? 0.65 : 0; float sqr = (s.patchRam.switch1 & 0x08) ? 0.63 : 0; //? 0.175 : 0;
float saw = (s.patchRam.switch1 & 0x10) ? 0.83 : 0; float saw = (s.patchRam.switch1 & 0x10) ? 0.8 : 0; //? 0.220 : 0;
float sub = (s.patchRam.sub / 127.0f); float sub = (s.patchRam.sub / 127.0f); // * 0.275;
float noise = (s.patchRam.noise / 127.0f) * 0.8;
float gain = 0.5 * powf(2, (s.patchRam.vca / 64.0f) - 1);
float gain = 0.5*powf(2, (4*s.patchRam.vca / 127.0f) - 1.0f);
//printf("%f %f\n",s.patchRam.vca / 127.0f, gain);
float vcaEnv = (s.patchRam.switch2 & 0x04) ? (float)ff11 : (env / 16384.0f); float vcaEnv = (s.patchRam.switch2 & 0x04) ? (float)ff11 : (env / 16384.0f);
@ -102,11 +98,6 @@ void Voice::run(Synth &s, float *buffer, uint32_t samples) {
// out = (2 * y) - 1; // out = (2 * y) - 1;
out = y; out = y;
lcg = (lcg*519)+3;
out += noise * ((lcg / 2147483647.0f)-1);
// widthDelay = pw; // widthDelay = pw;
out *= 0.707; out *= 0.707;
@ -127,8 +118,7 @@ void Voice::run(Synth &s, float *buffer, uint32_t samples) {
vr58c106 += ((vcaEnv - vr58c106) * 0.005); vr58c106 += ((vcaEnv - vr58c106) * 0.005);
lastpw = pw; lastpw = pw;
// gain for fully resonant needs to be 128mV output out = b4 * (0.25);
out = b4 * (0.1625);
buffer[i] += (gain * out * vr58c106); buffer[i] += (gain * out * vr58c106);
} }

View File

@ -27,8 +27,6 @@ class Synth;
class Voice { class Voice {
public: public:
uint8_t note = 60; // per-voice note, set to middle C at 02b1h uint8_t note = 60; // per-voice note, set to middle C at 02b1h
float detune[8] = {-0.13, -0.09, -0.05, -0.025, 0.018, 0.06, 0.11, 0.16};
uint8_t voicenum;
void on(uint32_t key, bool reset); void on(uint32_t key, bool reset);
void off(); void off();
@ -42,8 +40,6 @@ class Voice {
float omega; float omega;
uint16_t vcfenv; uint16_t vcfenv;
float phase = 0, subosc = 1;
private: private:
enum { ATTACK, enum { ATTACK,
@ -65,10 +61,10 @@ class Voice {
uint16_t env; uint16_t env;
float phase = 0, subosc = 1;
// float env, target; // float env, target;
float delay; float delay;
uint8_t pulseStage = 0; uint8_t pulseStage = 0;
uint32_t lcg;
float lastpw = 0; float lastpw = 0;
float vr58c106 = 0; float vr58c106 = 0;
@ -83,12 +79,6 @@ class Synth {
uint32_t blockLeft; uint32_t blockLeft;
uint32_t framesLeft = 0; uint32_t framesLeft = 0;
uint8_t polymode;
uint8_t noteTbl[8];
uint8_t voiceTbl[8];
float unisonDetune;
bool keyon; bool keyon;
uint8_t ff63 = 0; uint8_t ff63 = 0;
@ -143,7 +133,6 @@ class Synth {
} patchRam; } patchRam;
float pitchCV[104]; float pitchCV[104];
float hptmp;
void runLFO(); void runLFO();
void lfoDelay(); void lfoDelay();

View File

@ -296,10 +296,6 @@ h0407:
h0432: h0432:
// printf("setting omega for note %d \n", a); // printf("setting omega for note %d \n", a);
omega = ((s.pitchCV[a + 1] - s.pitchCV[a]) * (s.ff6e / 256.0)) + s.pitchCV[a]; omega = ((s.pitchCV[a + 1] - s.pitchCV[a]) * (s.ff6e / 256.0)) + s.pitchCV[a];
omega *= 1 + (s.unisonDetune*0.25*detune[voicenum]);
//printf("%f %f %d\n", s.unisonDetune, omega, voicenum);
//phase = (2.4*omega*voicenum);
// 0432 onwards calculates the address for the CV // 0432 onwards calculates the address for the CV
// table at E60 and stacks it // table at E60 and stacks it
// 043a onwards fetches the value from the divider // 043a onwards fetches the value from the divider