This effect was discovered by error, playing with some counters I end up with this lo-fi code that sounds like this guitar riff in "Television Rules the Nation" by Daft Punk.
Pressing the push-buttons you can octave the sound up or down.
How does it work? I still don't know, basically, it messes with the sampling frequency of the output signal that creates this tones
This is the complete code:
// www.Electrosmash.com/pedalshield-mega
/* pedalshield_mega_daftpunk_distortion_octaver.ino creates a lo-fi distortion effect similar to the used
* in Television Rules the Nation by Daft Punk. It also octaves the signal up/down presing the pushbuttons
*/
#include "U8glib.h"
U8GLIB_SH1106_128X64 u8g(U8G_I2C_OPT_NO_ACK); // Display which does not send ACK
//defining harware 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
//other variables
int counter = 0;
int counter2 = 0;
int input;
int dist_variable=10;
byte ADC_low, ADC_high;
void setup() {
//setup IO
pinMode(FOOTSWITCH, INPUT_PULLUP);
pinMode(PUSHBUTTON_1, INPUT_PULLUP);
pinMode(PUSHBUTTON_2, INPUT_PULLUP);
pinMode(LED, OUTPUT);
pinMode(6, OUTPUT); //PWM0 as output
pinMode(7, OUTPUT); //PWM1 as output
// setup ADC
ADMUX = 0x60; // left adjust, adc0, internal vcc
ADCSRA = 0xe5; // turn on adc, ck/32, auto trigger
ADCSRB = 0x00; // ADC free running mode
DIDR0 = 0x01; // turn off digital inputs for adc0
// setup PWM
TCCR4A = (((PWM_QTY - 1) << 5) | 0x80 | (PWM_MODE << 1)); //
TCCR4B = ((PWM_MODE << 3) | 0x11); // ck/1
TIMSK4 = 0x20; // interrupt on capture interrupt
ICR4H = (PWM_FREQ >> 8);
ICR4L = (PWM_FREQ & 0xff);
DDRB |= ((PWM_QTY << 1) | 0x02); // turn on outputs
sei(); // turn on interrupts - not really necessary with arduino
}
void loop()
{
//Turn on the LED and write the OLED if the effect is ON.
if (digitalRead(FOOTSWITCH))
{
digitalWrite(LED, HIGH); //light the LED
u8g.firstPage();
do {
u8g.setFont(u8g_font_helvR14r);
u8g.drawStr( 0, 16, " DAFT PUNK ");
u8g.drawStr( 0, 36, " OCTAVER ");
u8g.setPrintPos(45, 64);
u8g.setFont(u8g_font_helvR24r);
u8g.print(dist_variable);
} while( u8g.nextPage() );
}
else
{
digitalWrite(LED, LOW); // switch-off the LED
u8g.firstPage();
do {
u8g.setFont(u8g_font_helvR24r);
u8g.drawStr( 0, 32, "EFFECT ");
u8g.drawStr( 0, 64, " OFF ");
} while( u8g.nextPage() );
}
}
ISR(TIMER4_CAPT_vect) {
counter++; //to save resources, the pushbuttons are checked every 2000 times.
if(counter==2000)
{
counter=0;
if (!digitalRead(PUSHBUTTON_1)) {
if (dist_variable<500)dist_variable++;
digitalWrite(LED, LOW); //blinks the led
}
if (!digitalRead(PUSHBUTTON_2)) {
if (dist_variable>0)dist_variable--;
digitalWrite(LED, LOW); //blinks the led
}
}
counter2++;
if(counter2>=dist_variable)
{
counter2=0;
// get ADC data
ADC_low = ADCL; // you need to fetch the low byte first
ADC_high = ADCH;
//construct the input sumple summing the ADC low and high byte.
input = ((ADC_high << 8) | ADC_low) + 0x8000; // make a signed 16b value
OCR4AL = ((input + 0x8000) >> 8); // convert to unsigned, send out high byte
OCR4BL = input; // send out low byte
}
}