Polyphonic MIDI to CV

diy
synthesis
electronics

May 22, 2021

After looking at the basics of analog synthesis in the last post, let's now look at how to approach polyphony.

So far, I used the Arturia Keystep to send pitch and gate CV to the VCO. To be able to play multiple notes at once, we need multiple channels of pitch and gate CV.

MIDI to CV

Using an Arduino + a DAC (= Digital to Analog Converter), we can set the pitch CV output to an analog voltage whenever a MIDI note message is received:

idea

The Arduino receives a MIDI note message, calculates the correct voltage for that note and sends the voltage through the DAC to the VCO.

Digital to Analog Conversion

Without a DAC, an Arudino is not able to produce true analog output™, meaning a voltage that is not either 0V or 5V.

What about analogWrite?

If you've already worked with Arduino, you might know the analogWrite function, which suggests that we output an analog voltage. But in fact, analogWrite will only output a PWM wave, which looks like this:

pwm waves

(this graphic looks bad on dark mode)

So this is still just a digital signal. It can be useful to dim LEDs for example, but it's not sufficient for a VCO.

Poor Mans DAC

What i heard somewhere referred to as the "Poor Mans DAC", is by filtering the PWM output of the arduino. This will filter out the higher frequencies of the pulse wave, leaving a more or less mean value. You can read more about it here.

I tested this too (using a TL074 instead of a TL2451):

poor mands dac

The problem: In my test, I could only get a voltage range from 1.7V to 4.4V, which only gives us 3 octaves (at 1V/octave) + the resulting voltage was very shaky.

The next thing I tried was a resistor ladder.

Resistor Ladder

We can use a bunch of resistors and arrange them to a so called resistor ladder to build a proper DAC:

resistor ladder

Essentially, we are using a multi layered Voltage Divider. The lower row of resistors are 2K and the upper row are 1K. To read more on how and why it works, I recommend this article.

Using this code:

void setup() {
  pinMode(5, OUTPUT);
  pinMode(6, OUTPUT);
  pinMode(7, OUTPUT);
  pinMode(8, OUTPUT);
  pinMode(9, OUTPUT);
  pinMode(10, OUTPUT);
  pinMode(11, OUTPUT);
  pinMode(12, OUTPUT);
}

void loop() {
  int value = 127; // value between 0 and 255
  int bits = 8;
  for (int i = 0; i < bits; ++i) {
    int digit = bitRead(value, i);
    digitalWrite(5 + i, digit);
  }
}

The value can be set between 0 and 255 (= 256 values = 2^8 = 8 bit DAC). If we measure the voltage, we get approximately 2.5V for 127:

resistor ladder

Let's step up the game once more and look at an even better solution, as we would quickly run out of output pins if we built our multi channel dac with resistor ladders.

Using the MCP4728

Using an IC DAC, like the MCP4728, we can have 4 channel true analog output™ without using zillions of pins. The integration is really easy with a breakout board, like the one from Adafruit:

mcp4728

Code:

// Basic demo for configuring the MCP4728 4-Channel 12-bit I2C DAC
#include <Adafruit_MCP4728.h>
#include <Wire.h>

Adafruit_MCP4728 mcp;

void setup(void) {
  Serial.begin(115200);
  while (!Serial)
    delay(10); // will pause Zero, Leonardo, etc until serial console opens

  Serial.println("Adafruit MCP4728 test!");

  // Try to initialize!
  if (!mcp.begin()) {
    Serial.println("Failed to find MCP4728 chip");
    while (1) {
      delay(10);
    }
  }

  mcp.setChannelValue(MCP4728_CHANNEL_A, 4095);
  mcp.setChannelValue(MCP4728_CHANNEL_B, 2048);
  mcp.setChannelValue(MCP4728_CHANNEL_C, 1024);
  mcp.setChannelValue(MCP4728_CHANNEL_D, 0);
}

void loop() { delay(1000); }

Reading Channel B:

mcp4728 test

For a full tutorial see this page.

Handling MIDI Messages

Now that we have 4 12 bit output channels, we can think about how to translate a midi message to voltage. To simplify MIDI handling, we can use the MIDIUSB library.

Taken from this example, we can read incoming midi messages like so:

#include "MIDIUSB.h"
#include <Adafruit_MCP4728.h>
#include <Wire.h>

void loop() {
  midiEventPacket_t rx;
  rx = MidiUSB.read();

  switch (rx.header) {
    case 0:
      break; //No pending events

    case 0x9:
      noteOn(
        rx.byte1 & 0xF,  //channel
        rx.byte2,        //pitch
        rx.byte3         //velocity
      );
      break;

    case 0x8:
      noteOff(
        rx.byte1 & 0xF,  //channel
        rx.byte2,        //pitch
        rx.byte3         //velocity
      );
      break;

    case 0xB:
      controlChange(
        rx.byte1 & 0xF,  //channel
        rx.byte2,        //control
        rx.byte3         //value
      );
      break;
  }
}

MIDI note to Voltage

In the noteOn method, we receive the pitch as MIDI note number:

int midiOffset = 24;

int getStep(byte pitch) {
  int steps = 4095;
  int octaves = 5;
  return max(0, min(round(steps / (octaves * 12) * (pitch - midiOffset)), steps));
}

void noteOn(byte midiChannel, byte pitch, byte velocity) {
  mcp.setChannelValue(MCP4728_CHANNEL_A, getStep(pitch));
}

// rest of code in last snippet

Now we can use the output channel A to control a VCO via MIDI!

Make it polyphonic!

Using the vco micromodules introduced in the last post, we can create a 3 oscillator setup like this:

3vco simple

  • Channels A, B and C of the DAC go the the VCOs 1V/oct inputs
  • Pots allow tuning the oscillators
  • The saw outputs of the oscillators are mixed together using the mix module
  • Digital Pins 4, 5, 6 of the Arduino are connected to the gate of the mix mixodule

Voice Allocation

As we have only 3 oscillators, we need some logic that allocates each of them to incoming notes. To do that, we need the following variables:

int channels = 3;
int pressed[] = { -1, -1, -1}; // which notes are allocated on which channel, -1 = none
int gates[] = { 6, 5, 4}; // which pins correspond to which channel gate

In the noteOn method, we can iterate over the pressed array and allocate the first unused channel to the incoming note:

void noteOn(byte midiChannel, byte pitch, byte velocity) {
  int channel = -1;
  for (int i = 0; i < channels; i++) {
    if (pressed[i] == -1 && channel == -1) {
      channel = i;
      pressed[i] = pitch;
      digitalWrite(gates[channel], HIGH);
      mcp.setChannelValue(channel, getStep(pitch));
    }
  }
}

When the note is released, we have to revert what we did:

void noteOff(byte midiChannel, byte pitch, byte velocity) {
  int channel = -1;
  for (int i = 0; i < channels; i++) {
    if (pressed[i] == pitch && channel == -1) {
      channel = i;
      digitalWrite(gates[channel], LOW);
      pressed[channel] = -1;
    }
  }
}

And that's all we need for polyphonic CV messages! This is the starting point to build any paraphonic / polyphonic synth. In the next post, I will expand the above synth to a paraphonic poor mans prophet.

Bonus: Is 12 bit enough for pitch CV?

Some might think now: Is 12 bit anough for true analog output™ pitch CV? Let's calculate a little bit..

  • With 12 bit, we have 2^12 = 4096 different steps representing voltages from 0 to 5V.
  • At 1V/oct, we have 5*12 = 60 semitones in the volt range.
  • Each semitone can be broken down to 100 cents, so we have 6000 cents in our range, giving us 6000/4096 = 1.4 cents per step.
  • According to this page, the so called "Just Noticeable Difference" (JND) in pitch perception is around 5 cents.
  • Having 1.4 cents per step, we are way below the JND.

So 12 bit should be fine.. (Maybe smaller changes in pitch are perceptible in the context of chords, but that might be topic of another discussion).

Future Ideas

Similar Devices

Felix Roos 2022