264 lines
No EOL
5 KiB
C++
264 lines
No EOL
5 KiB
C++
// RING BUFFER VARIABLE PITCH SHIFT
|
|
// via www.Electrosmash.com
|
|
// via OpenMusicLabs
|
|
|
|
//defining hardware resources.
|
|
#define LED 13
|
|
#define FOOTSWITCH 12
|
|
#define TOGGLE 2
|
|
#define PUSHBUTTON_1 A5
|
|
#define PUSHBUTTON_2 A4
|
|
|
|
//defining the output PWM parameters
|
|
#define PWM_FREQ 0x00FF // pwm frequency - 31.3KHz
|
|
#define PWM_MODE 0 // Fast (1) or Phase Correct (0)
|
|
#define PWM_QTY 2 // 2 PWMs in parallel
|
|
#define BUFF_SIZE 500 // size of ring buffer, can theoretically be anything
|
|
// 500 should hold 15.9 ms of audio
|
|
//
|
|
|
|
// STRUCTURES
|
|
|
|
struct button_time_s
|
|
{
|
|
long footswitch;
|
|
long pb1;
|
|
long pb2;
|
|
};
|
|
|
|
struct count_s
|
|
{
|
|
int main;
|
|
int interrupt;
|
|
};
|
|
|
|
struct switch_flag_s
|
|
{
|
|
bool current;
|
|
bool old;
|
|
};
|
|
|
|
struct ring_buff_s
|
|
{
|
|
int buff[BUFF_SIZE];
|
|
int head;
|
|
int tail;
|
|
};
|
|
|
|
struct tempo_s
|
|
{
|
|
int interval;
|
|
long timeout;
|
|
};
|
|
|
|
struct divider_s
|
|
{
|
|
int input;
|
|
int output;
|
|
};
|
|
|
|
// VARIABLES
|
|
int input, vol_variable=512;
|
|
byte ADC_low, ADC_high;
|
|
bool led_flag;
|
|
int offset = 5;
|
|
long milli;
|
|
button_time_s button_time;
|
|
count_s count;
|
|
switch_flag_s switch_flag;
|
|
ring_buff_s buf;
|
|
divider_s divider;
|
|
tempo_s tempo;
|
|
|
|
// BUFFER FUNCTIONS
|
|
void push_to_buff(int in)
|
|
{
|
|
buf.buff[buf.head] = in;
|
|
buf.head++;
|
|
if (buf.head == BUFF_SIZE)
|
|
{
|
|
buf.head = 0;
|
|
}
|
|
}
|
|
|
|
int pop_from_buff (void)
|
|
{
|
|
int out = buf.buff[buf.tail];
|
|
if (buf.tail != buf.head)
|
|
{
|
|
buf.tail++;
|
|
}
|
|
else
|
|
{
|
|
buf.buff[buf.tail]--;
|
|
}
|
|
if (buf.tail == BUFF_SIZE)
|
|
{
|
|
buf.tail = 0;
|
|
}
|
|
return out;
|
|
}
|
|
|
|
void setup()
|
|
{
|
|
//setup IO
|
|
Serial.begin(9600);
|
|
pinMode(FOOTSWITCH, INPUT_PULLUP);
|
|
pinMode(PUSHBUTTON_1, INPUT_PULLUP);
|
|
pinMode(PUSHBUTTON_2, INPUT_PULLUP);
|
|
pinMode(LED, OUTPUT);
|
|
|
|
// setup ADC
|
|
ADMUX = 0x60; // left adjust, adc0, internal vcc
|
|
ADCSRA = 0xe5; // turn on adc, ck/32, auto trigger
|
|
ADCSRB = 0x07; // t1 capture for trigger
|
|
DIDR0 = 0x01; // turn off digital inputs for adc0
|
|
|
|
// setup PWM
|
|
TCCR1A = (((PWM_QTY - 1) << 5) | 0x80 | (PWM_MODE << 1)); //
|
|
TCCR1B = ((PWM_MODE << 3) | 0x11); // ck/1
|
|
TIMSK1 = 0x20; // interrupt on capture interrupt
|
|
ICR1H = (PWM_FREQ >> 8);
|
|
ICR1L = (PWM_FREQ & 0xff);
|
|
DDRB |= ((PWM_QTY << 1) | 0x02); // turn on outputs
|
|
sei(); // turn on interrupts - not really necessary with arduino
|
|
tempo.interval = 150;
|
|
divider.input = 0;
|
|
divider.output = 9;
|
|
}
|
|
|
|
void tempo_setter()
|
|
{
|
|
if (milli > tempo.timeout)
|
|
{
|
|
led_flag = !led_flag;
|
|
digitalWrite(LED, led_flag);
|
|
tempo.timeout = milli + tempo.interval;
|
|
}
|
|
}
|
|
|
|
void tempo_getter()
|
|
{
|
|
if (digitalRead(PUSHBUTTON_1) == false)
|
|
{
|
|
if (milli > button_time.pb1)
|
|
{
|
|
if (milli - button_time.pb1 > 150)
|
|
{
|
|
tempo.interval = milli - button_time.pb1;
|
|
tempo.timeout = milli + tempo.interval;
|
|
}
|
|
button_time.pb1 = milli;
|
|
}
|
|
}
|
|
}
|
|
|
|
void pitch_getter()
|
|
{
|
|
if (digitalRead(PUSHBUTTON_2) == false)
|
|
{
|
|
if (milli - button_time.pb2 > 150)
|
|
{
|
|
divider.output++;
|
|
if (divider.output > 15)
|
|
{
|
|
divider.output = 0;
|
|
}
|
|
if (divider.output == 1)
|
|
{
|
|
divider.output = 2;
|
|
}
|
|
button_time.pb2 = milli;
|
|
}
|
|
}
|
|
}
|
|
|
|
// MAIN LOOP
|
|
// checks the footswitch is pushed every 100 loops
|
|
// changes the switch_flag
|
|
// if switch flag is true call functions
|
|
// else just set LED here,
|
|
// doesn't need to optimised if no audio is being processed
|
|
// increment count
|
|
|
|
void loop()
|
|
{
|
|
if (count.main % 100 == 0)
|
|
{
|
|
switch_flag.current = digitalRead(FOOTSWITCH);
|
|
if (switch_flag.current != switch_flag.old)
|
|
{
|
|
button_time.footswitch = millis();
|
|
digitalWrite(LED, true);
|
|
Serial.println(button_time.footswitch);
|
|
}
|
|
switch_flag.old = switch_flag.current;
|
|
if (switch_flag.current == true)
|
|
{
|
|
milli = millis();
|
|
pitch_getter();
|
|
tempo_getter();
|
|
tempo_setter();
|
|
}
|
|
else
|
|
{
|
|
digitalWrite(LED, false);
|
|
}
|
|
}
|
|
count.main++;
|
|
}
|
|
|
|
// AUDIO INPUT
|
|
// pulls the audio from the ADC
|
|
// pushes it to the ring buffer
|
|
void audio_inputter(int divide)
|
|
{
|
|
ADC_low = ADCL; // you need to fetch the low byte first
|
|
ADC_high = ADCH;
|
|
input = ((ADC_high << 8) | ADC_low); // make a signed 16b value
|
|
if (input > 0x8000)
|
|
{
|
|
input = input + 0x8000;
|
|
}
|
|
push_to_buff(input);
|
|
}
|
|
|
|
// AUDIO OUTPUTTER
|
|
// if the eqn is != 0
|
|
// pop audio from ring buffer and output to PWM
|
|
void audio_outputter(int divide)
|
|
{
|
|
if (milli - button_time.footswitch > offset)
|
|
{
|
|
// Serial.println("YE");
|
|
if (count.interrupt % divide != 0)
|
|
{
|
|
int output = pop_from_buff();
|
|
OCR1AL = ((output + 0x8000) >> 8); // convert to unsigned, send out high byte
|
|
OCR1BL = output; // send out low byt
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
// TIMER1 INTERRUPT
|
|
// checks the switch flag is true
|
|
// if LED flag is true, apply effect
|
|
// else do not apply effect
|
|
ISR(TIMER1_CAPT_vect)
|
|
{
|
|
if (switch_flag.current == true)
|
|
{
|
|
if (led_flag == true)
|
|
{
|
|
audio_inputter(divider.input);
|
|
audio_outputter(divider.output);
|
|
}
|
|
else
|
|
{
|
|
audio_inputter(0);
|
|
audio_outputter(0);
|
|
}
|
|
count.interrupt++;
|
|
}
|
|
} |