Files
2022-07-21 15:35:09 +02:00

254 lines
7.8 KiB
C++

/*
* Tap Tremelo Pedal Firmware
* V1 by Dennis Gunia
*
* 2022/07
*
*/
#define lmicros() ((long)micros())
//Lookup Tables
static const uint8_t sineLUT[256] PROGMEM = {
0x80, 0x83, 0x86, 0x89, 0x8c, 0x8f, 0x92, 0x95, 0x98, 0x9b, 0x9e, 0xa2, 0xa5, 0xa7, 0xaa, 0xad,
0xb0, 0xb3, 0xb6, 0xb9, 0xbc, 0xbe, 0xc1, 0xc4, 0xc6, 0xc9, 0xcb, 0xce, 0xd0, 0xd3, 0xd5, 0xd7,
0xda, 0xdc, 0xde, 0xe0, 0xe2, 0xe4, 0xe6, 0xe8, 0xea, 0xeb, 0xed, 0xee, 0xf0, 0xf1, 0xf3, 0xf4,
0xf5, 0xf6, 0xf8, 0xf9, 0xfa, 0xfa, 0xfb, 0xfc, 0xfd, 0xfd, 0xfe, 0xfe, 0xfe, 0xff, 0xff, 0xff,
0xff, 0xff, 0xff, 0xff, 0xfe, 0xfe, 0xfe, 0xfd, 0xfd, 0xfc, 0xfb, 0xfa, 0xfa, 0xf9, 0xf8, 0xf6,
0xf5, 0xf4, 0xf3, 0xf1, 0xf0, 0xee, 0xed, 0xeb, 0xea, 0xe8, 0xe6, 0xe4, 0xe2, 0xe0, 0xde, 0xdc,
0xda, 0xd7, 0xd5, 0xd3, 0xd0, 0xce, 0xcb, 0xc9, 0xc6, 0xc4, 0xc1, 0xbe, 0xbc, 0xb9, 0xb6, 0xb3,
0xb0, 0xad, 0xaa, 0xa7, 0xa5, 0xa2, 0x9e, 0x9b, 0x98, 0x95, 0x92, 0x8f, 0x8c, 0x89, 0x86, 0x83,
0x80, 0x7c, 0x79, 0x76, 0x73, 0x70, 0x6d, 0x6a, 0x67, 0x64, 0x61, 0x5d, 0x5a, 0x58, 0x55, 0x52,
0x4f, 0x4c, 0x49, 0x46, 0x43, 0x41, 0x3e, 0x3b, 0x39, 0x36, 0x34, 0x31, 0x2f, 0x2c, 0x2a, 0x28,
0x25, 0x23, 0x21, 0x1f, 0x1d, 0x1b, 0x19, 0x17, 0x15, 0x14, 0x12, 0x11, 0xf, 0xe, 0xc, 0xb,
0xa, 0x9, 0x7, 0x6, 0x5, 0x5, 0x4, 0x3, 0x2, 0x2, 0x1, 0x1, 0x1, 0x0, 0x0, 0x0,
0x0, 0x0, 0x0, 0x0, 0x1, 0x1, 0x1, 0x2, 0x2, 0x3, 0x4, 0x5, 0x5, 0x6, 0x7, 0x9,
0xa, 0xb, 0xc, 0xe, 0xf, 0x11, 0x12, 0x14, 0x15, 0x17, 0x19, 0x1b, 0x1d, 0x1f, 0x21, 0x23,
0x25, 0x28, 0x2a, 0x2c, 0x2f, 0x31, 0x34, 0x36, 0x39, 0x3b, 0x3e, 0x41, 0x43, 0x46, 0x49, 0x4c,
0x4f, 0x52, 0x55, 0x58, 0x5a, 0x5d, 0x61, 0x64, 0x67, 0x6a, 0x6d, 0x70, 0x73, 0x76, 0x79, 0x7c
};
static const uint8_t bumpLUT[256] PROGMEM = {
0, 4, 7, 10, 13, 16, 19, 22, 25, 29, 32, 35, 38, 41, 44, 47,
50, 53, 56, 59, 62, 65, 69, 72, 75, 78, 80, 83, 86, 89, 92, 95,
98, 101, 104, 107, 110, 112, 115, 118, 121, 123, 126, 129, 132, 134, 137, 140,
142, 145, 147, 150, 152, 155, 157, 160, 162, 165, 167, 169, 172, 174, 176, 179,
181, 183, 185, 187, 189, 192, 194, 196, 198, 200, 202, 203, 205, 207, 209, 211,
213, 214, 216, 218, 219, 221, 222, 224, 225, 227, 228, 230, 231, 232, 234, 235,
236, 237, 238, 240, 241, 242, 243, 244, 245, 245, 246, 247, 248, 249, 249, 250,
251, 251, 252, 252, 253, 253, 254, 254, 254, 255, 255, 255, 255, 255, 255, 255,
255, 255, 255, 255, 255, 255, 255, 255, 254, 254, 254, 253, 253, 252, 252, 251,
251, 250, 249, 249, 248, 247, 246, 245, 245, 244, 243, 242, 241, 240, 238, 237,
236, 235, 234, 232, 231, 230, 228, 227, 225, 224, 222, 221, 219, 218, 216, 214,
213, 211, 209, 207, 205, 203, 202, 200, 198, 196, 194, 192, 189, 187, 185, 183,
181, 179, 176, 174, 172, 169, 167, 165, 162, 160, 157, 155, 152, 150, 147, 145,
142, 140, 137, 134, 132, 129, 126, 123, 121, 118, 115, 112, 110, 107, 104, 101,
98, 95, 92, 89, 86, 83, 80, 78, 75, 72, 69, 65, 62, 59, 56, 53,
50, 47, 44, 41, 38, 35, 32, 29, 25, 22, 19, 16, 13, 10, 7, 4
};
//Rate Modifier Ranges
/* 0
* 114
* 228
* 341
* 512
* 683
* 796
* 910
* 1024
*/
#define RM_NOTCHES_WIDTH 48
#define RM_N0_MID 0
#define RM_N1_MID 128
#define RM_N2_MID 256
#define RM_N3_MID 384
#define RM_N4_MID 512
#define RM_N5_MID 640
#define RM_N6_MID 768
#define RM_N7_MID 896
#define RM_N8_MID 1024
//Waveform Constants
#define WF_SAW 1
#define WF_SINE 2
#define WF_SQUARE 4
#define TS_TAP 0
#define TS_POT 1
int depth = 0;
unsigned long delayTime = 4000;
unsigned long delayTimeOUT = 4000;
uint8_t waveform = WF_SINE;
uint8_t tempoSource = TS_TAP;
uint8_t symetry = 127;
double modifier = 4;
uint16_t wfStep = 0;
int potTempoLast = 0;
double bpm = 60;
unsigned long lastStep = lmicros();
unsigned long lastTest = lmicros();
unsigned long lastTap = 0;
//IO Pin Definition
#define PIN_A_RATE A0 //Pot Rate
#define PIN_A_DEPTH A1 //Pot Depth
#define PIN_A_SHAPE A2 //Pot Shape
#define PIN_A_MUL A3 //Pot Clock-Modifier
#define PIN_D_WF1 10 //Waveform Switch Pole 1
#define PIN_D_WF2 7 //Waveform Switch Pole 2
#define PIN_D_TAP 9 //Tap Tempo Push button
#define PIN_D_MOD 8 //Tempo Modulation Switch
#define PIN_PWM A5
void setup() {
// setup Timer 1 to 31.2kHz @ 8 Bit
TCCR1B &= 0b11100000;
TCCR1B |= 0b00001001;
TCCR1A &= 0b11111100;
TCCR1A |= 0b00000001;
//Set input pins to pullup
pinMode(PIN_D_WF1, INPUT_PULLUP);
pinMode(PIN_D_WF2, INPUT_PULLUP);
pinMode(PIN_D_TAP, INPUT_PULLUP);
pinMode(PIN_D_MOD, INPUT_PULLUP);
//enable pin change interrupt
GIMSK = 0b00100000;
PCMSK1 = 0b00000010;
MCUCR &= 0b00000011;
MCUCR |= 0b00000010;
}
uint8_t pstate = 0;
ISR(PCINT1_vect) {
if (lastTap > lmicros() - 800000 ){ //debounce
return;
}
if (pstate == 0){
pstate = 1;
}else{
pstate = 0;
return;
}
//delayTime = 500;
if (lastTap == 0){
lastTap = lmicros();
}else{
delayTime = (lmicros() - lastTap)/256;
lastTap = lmicros();
wfStep = 0;
tempoSource = TS_TAP;
}
}
//Read pots and switch state
void getInput() {
depth = 255 - (analogRead(PIN_A_DEPTH) / 4);
//delayTime = map(analogRead(A1),0,1023,5000,20000);
int raw = analogRead(PIN_A_RATE);
if (tempoSource == TS_POT){
delayTime = map(raw, 0, 1023, 0, 40000) * 8;
potTempoLast = raw;
}else{
if (potTempoLast < (raw - 16)) {tempoSource = TS_POT;}
if (potTempoLast > (raw + 16)) {tempoSource = TS_POT;}
}
symetry = map(analogRead(PIN_A_SHAPE), 0, 1023, 4, 251); ;
//float secFracion = 1/((float)bpm*256);
//delayTime = secFracion*10000000;
waveform = WF_SAW;
if (digitalRead(PIN_D_WF1) == LOW) {
waveform = WF_SQUARE;
}
if (digitalRead(PIN_D_WF2) == LOW) {
waveform = WF_SINE;
}
//rate modifier
raw = analogRead(PIN_A_MUL);
if (raw >= 0 && raw <= RM_N0_MID + RM_NOTCHES_WIDTH) {modifier = 16;}
if (raw >= RM_N1_MID - RM_NOTCHES_WIDTH && raw <= RM_N1_MID + RM_NOTCHES_WIDTH) {modifier = 8;}
if (raw >= RM_N2_MID - RM_NOTCHES_WIDTH && raw <= RM_N2_MID + RM_NOTCHES_WIDTH) {modifier = 4;}
if (raw >= RM_N3_MID - RM_NOTCHES_WIDTH && raw <= RM_N3_MID + RM_NOTCHES_WIDTH) {modifier = 2;}
if (raw >= RM_N4_MID - RM_NOTCHES_WIDTH && raw <= RM_N4_MID + RM_NOTCHES_WIDTH) {modifier = 1;}
if (raw >= RM_N5_MID - RM_NOTCHES_WIDTH && raw <= RM_N5_MID + RM_NOTCHES_WIDTH) {modifier = 0.5;}
if (raw >= RM_N6_MID - RM_NOTCHES_WIDTH && raw <= RM_N6_MID + RM_NOTCHES_WIDTH) {modifier = 0.25;}
if (raw >= RM_N7_MID - RM_NOTCHES_WIDTH && raw <= RM_N7_MID + RM_NOTCHES_WIDTH) {modifier = 0.125;}
if (raw >= RM_N8_MID && raw <= 1024) {modifier = 0.0625;}
applySppedMod();
}
void applySppedMod(){
if (digitalRead(PIN_D_MOD) == HIGH) {
delayTimeOUT = delayTime*modifier;
}else{
delayTimeOUT = delayTime;
}
}
void loop() {
unsigned long millisNow = lmicros() ; //Timestamp NOW
if (lastTest < millisNow - 800) { //Get panel state every 100ms
lastTest = millisNow;
getInput();
}
if (lastStep < millisNow - delayTimeOUT) { //do waveform stuff
wfStep ++;
if (wfStep > 255) wfStep = 0;
lastStep = millisNow;
if ( waveform == WF_SAW ) {
// SAW/TRI
if (wfStep < symetry) {
analogWrite(PIN_PWM, map(wfStep, 0, symetry, depth, 255));
} else {
analogWrite(PIN_PWM, map(wfStep, 255, symetry, depth, 255));
}
} else if ( waveform == WF_SINE ) {
// SINE/BUMP
if ( symetry >= 63 && symetry <= 191){
uint8_t val = pgm_read_word_near(sineLUT + (wfStep));
analogWrite(PIN_PWM, map(val, 0, 255, depth, 255));
}else{
uint8_t val = pgm_read_word_near(bumpLUT + (wfStep));
if (symetry > 127){val = 255-val;}
analogWrite(PIN_PWM, map(val, 0, 255, depth, 255));
}
} else {
// SQUARE/PWM
if (wfStep < symetry) {
analogWrite(PIN_PWM, depth);
} else {
analogWrite(PIN_PWM, 255);
}
}
}
}