249 lines
5.3 KiB
C++
249 lines
5.3 KiB
C++
#include <Adafruit_GFX.h>
|
|
#include <Adafruit_MPR121.h>
|
|
#include <Adafruit_SSD1306.h>
|
|
#include <Audio.h>
|
|
#include <Fonts/FreeSans9pt7b.h>
|
|
#include <Wire.h>
|
|
#include <paj7620.h>
|
|
#include <stdio.h>
|
|
|
|
#include "patches.h"
|
|
#include "pipe.h"
|
|
#include "synth.h"
|
|
#include "tuning.h"
|
|
|
|
const char *buildDate = __DATE__;
|
|
|
|
#if defined(ADAFRUIT_TRELLIS_MAdafruit_SSD1306EXPRESS)
|
|
#include <Adafruit_NeoTrellisM4.h>
|
|
Adafruit_NeoTrellisM4 trellis; // = Adafruit_NeoTrellisM4();
|
|
#endif
|
|
|
|
Pipe pipe;
|
|
Tuning tuning = Tuning(NOTE_D4, PITCH_CONCERT_D4, TUNINGSYSTEM_JUST);
|
|
|
|
Adafruit_SSD1306 display(128, 32, &Wire, -1);
|
|
|
|
// Settings
|
|
uint8_t patch[4] = {0};
|
|
float volume[4] = {0};
|
|
|
|
// Pipes
|
|
#define NUM_DRONES 3
|
|
#define NUM_REGULATORS 3
|
|
FMVoice Chanter;
|
|
FMVoice Drones[NUM_DRONES];
|
|
FMVoice Regulators[NUM_REGULATORS];
|
|
|
|
AudioFilterBiquad biquad1;
|
|
AudioMixer4 mixDrones;
|
|
AudioMixer4 mixRegulators;
|
|
AudioMixer4 mixL;
|
|
AudioMixer4 mixR;
|
|
AudioSynthNoiseWhite noise;
|
|
|
|
#if defined(ADAFRUIT_TRELLIS_M4_EXPRESS)
|
|
AudioOutputAnalogStereo out1;
|
|
#else
|
|
AudioOutputI2S out1;
|
|
#endif
|
|
|
|
AudioControlSGTL5000 sgtl5000;
|
|
|
|
AudioConnection FMVoicePatchCords[] = {
|
|
{noise, 0, mixL, 3},
|
|
{noise, 0, mixR, 3},
|
|
|
|
{Chanter.outputMixer, 0, biquad1, 0},
|
|
{biquad1, 0, mixL, 0},
|
|
{biquad1, 0, mixR, 0},
|
|
|
|
{Drones[0].outputMixer, 0, mixDrones, 0},
|
|
{Drones[1].outputMixer, 0, mixDrones, 1},
|
|
{Drones[2].outputMixer, 0, mixDrones, 2},
|
|
{mixDrones, 0, mixL, 1},
|
|
{mixDrones, 0, mixR, 1},
|
|
|
|
{Regulators[0].outputMixer, 0, mixRegulators, 0},
|
|
{Regulators[1].outputMixer, 0, mixRegulators, 1},
|
|
{Regulators[2].outputMixer, 0, mixRegulators, 2},
|
|
{mixRegulators, 0, mixL, 2},
|
|
{mixRegulators, 0, mixR, 2},
|
|
|
|
{mixL, 0, out1, 0},
|
|
{mixR, 0, out1, 1},
|
|
|
|
FMVoiceWiring(Chanter),
|
|
FMVoiceWiring(Drones[0]),
|
|
FMVoiceWiring(Drones[1]),
|
|
FMVoiceWiring(Drones[2]),
|
|
FMVoiceWiring(Regulators[0]),
|
|
FMVoiceWiring(Regulators[1]),
|
|
FMVoiceWiring(Regulators[2]),
|
|
};
|
|
|
|
void blink(bool forever) {
|
|
for (;;) {
|
|
digitalWrite(LED_BUILTIN, true);
|
|
delay(200);
|
|
digitalWrite(LED_BUILTIN, false);
|
|
delay(200);
|
|
if (!forever) {
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
void diag(const char *fmt, ...) {
|
|
va_list args;
|
|
char s[80];
|
|
|
|
va_start(args, fmt);
|
|
vsnprintf(s, sizeof(s) - 1, fmt, args);
|
|
va_end(args);
|
|
|
|
display.clearDisplay();
|
|
display.drawRect(124, 16, 4, 16, SSD1306_WHITE);
|
|
display.setTextColor(SSD1306_WHITE);
|
|
display.setFont();
|
|
display.setTextSize(1);
|
|
|
|
display.setCursor(56, 24);
|
|
display.print(buildDate);
|
|
|
|
#if 0
|
|
display.setCursor(0, 16);
|
|
display.print(fn);
|
|
display.print(":");
|
|
display.print(lineno);
|
|
#endif
|
|
|
|
display.setCursor(0, 0);
|
|
display.print(s);
|
|
|
|
display.display();
|
|
}
|
|
|
|
// The right way to do this would be to make a Uilleann object,
|
|
// and pass that around.
|
|
// The Auido library makes this sort of a pain,
|
|
// and honestly, is anybody other than me going to use this?
|
|
#include "main-play.h"
|
|
#include "main-setup.h"
|
|
|
|
void setup() {
|
|
// Initialize settings
|
|
// XXX: Read these from persistent storage later
|
|
for (int i = 0; i < 4; i++) {
|
|
patch[i] = 0;
|
|
volume[i] = 0.75;
|
|
}
|
|
|
|
// PREPARE TO BLINK
|
|
pinMode(LED_BUILTIN, OUTPUT);
|
|
digitalWrite(LED_BUILTIN, true);
|
|
|
|
// Set up I2C. Apparently this needs a bit of startup delay.
|
|
Wire.begin();
|
|
|
|
// Initialize display
|
|
if (!display.begin(SSD1306_SWITCHCAPVCC, 0x3c)) {
|
|
blink(true);
|
|
}
|
|
digitalWrite(LED_BUILTIN, false);
|
|
diag("Hello!");
|
|
|
|
#if defined(ADAFRUIT_TRELLIS_M4_EXPRESS)
|
|
diag("Trellis...");
|
|
trellis.begin();
|
|
#endif
|
|
|
|
diag("Pipe...");
|
|
while (!pipe.Init()) {
|
|
diag("No pipe. Is it connected?");
|
|
blink(false);
|
|
}
|
|
|
|
diag("Audio...");
|
|
AudioMemory(20);
|
|
AudioProcessorUsageMaxReset();
|
|
AudioMemoryUsageMaxReset();
|
|
sgtl5000.enable();
|
|
sgtl5000.volume(0.3);
|
|
|
|
diag("Synth...");
|
|
loadPatch(0);
|
|
loadPatch(1);
|
|
loadPatch(2);
|
|
|
|
diag("Mixer...");
|
|
// Turn on all mixer channels
|
|
for (int i = 0; i < 4; i++) {
|
|
mixL.gain(i, volume[i]);
|
|
mixR.gain(i, volume[i]);
|
|
}
|
|
for (int i = 0; i < NUM_REGULATORS; ++i) {
|
|
mixRegulators.gain(i, 1);
|
|
}
|
|
for (int i = 0; i < NUM_DRONES; ++i) {
|
|
mixDrones.gain(i, 1);
|
|
}
|
|
biquad1.setBandpass(0, PITCH_CONCERT_A4, 1.0);
|
|
|
|
diag("Drones...");
|
|
playDrones();
|
|
|
|
diag("Done!");
|
|
display.dim(true);
|
|
}
|
|
|
|
void loadPatch(uint8_t where) {
|
|
FMPatch *p = &Bank[where];
|
|
|
|
switch (where) {
|
|
case 0:
|
|
Chanter.LoadPatch(p);
|
|
break;
|
|
case 1:
|
|
for (int i = 0; i < NUM_REGULATORS; ++i) {
|
|
Regulators[i].LoadPatch(p);
|
|
}
|
|
break;
|
|
case 2:
|
|
for (int i = 0; i < NUM_DRONES; ++i) {
|
|
Drones[i].LoadPatch(p);
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
void loop() {
|
|
static bool upSetting = true; // GET IT?
|
|
|
|
pipe.Update();
|
|
|
|
#if defined(ADAFRUIT_TRELLIS_M4_EXPRESS)
|
|
trellis.tick();
|
|
#endif
|
|
|
|
// If we're infinitely (for the sensor) off the knee,
|
|
// we might be in setup mode.
|
|
if (pipe.kneeClosedness == 0) {
|
|
// We only enter into setup mode if no keys are pressed.
|
|
// This hopefully avoids accidentally entering setup while playing.
|
|
// Like say you're playing a jaunty tune and suddenly there's an earthquake.
|
|
// You ought to be able to finish the tune off before your pipe goes into setup mode.
|
|
if (upSetting || (pipe.keys == 0)) {
|
|
doSetup();
|
|
upSetting = true;
|
|
return;
|
|
}
|
|
}
|
|
|
|
doPlay(upSetting);
|
|
upSetting = false;
|
|
}
|