Archive for August, 2010

Using OSC, Processing, and a Microcontroller to connect Max/MSP and a DAC IC.

My most recent project, the USB-Octomod, uses Processing to create an OpenSoundControl (OSC) interface between any OSC-ready software and a hardware DAC device I built. I’m going to break down the connections between different pieces of software and hardware, in order to explain how the system works and to provide the basis for a future tutorial on how one might use the device.

You can read more about the Octomod here, but it essentially allows computer control over the analog control voltages commonly used in analog synthesizers. Input a number 0 – 1023, and the device will output an analog voltage from -5V to +5V.

The OSC interface presents the inputs to the device, in the form of 8 numbered channels. The user sends an OSC message from their software of choice, and the interface program receives, processes, and communicates the data to the microcontroller in the device. I used a Teensy 2.0, which is very similar to Arduino, and any of this information should easily translate to the Arduino.

The OSC Interface

The OSC interface is simple. In your host program, you need to create a message formatted as follows:

/dac chanOne chanTwo chanThree chanFour chanFive chanSix chanSeven chanEight

For example:

/dac 256 273 50 1020 756 902 840 111

The trick here is that instead of sending an individual message whenever a channel changes, you can reduce network traffic by packaging all of the channels in one message, and updating that message at the rate of the most rapidly changing channel.

Sending OSC from Max/MSP

Here’s what it looks like in Max:

Screen shot 2010-08-25 at 8.55.57 AM

What is this code doing?

  1. The pak object outputs all eight of its inputs as a list whenever any one of the inputs changes.
  2. The message box below appends the /dac prefix to the list. Now the OSC message is formatted correctly.
  3. We don’t want the message to send automatically whenever a channel updates, so we buffer it with the second message box. This is done by sending the first message to the right inlet of the second message box.
  4. Finally, the metro object triggers the full OSC message to be sent once every 10ms.

The OSC interface application is expecting data on port 9999, and we’re going to be using the software locally, so we use the localhost address: 127.0.0.1. The Max udpsend object takes those two numbers as arguments, and transmits the OSC message.

Receiving OSC in Processing

The OSC interface program is written in Processing. OSC is easy to use in Processing as well. With a couple of lines of code, we’re ready to go:

import oscP5.*; // import the oscP5 library
import netP5.*; // the netP5 library is also required for the osc library
OscP5 oscP5;
oscP5 = new OscP5(this, 9999); // all you need to start oscp5 listening on port 9999

Now all we have to do is tell our program what to do when an OSC message is received. This is done by defining the oscEvent function.

After parsing out each of the eight input numbers, we check if a given channel needs to update its state. If so, we pass it to the writeValue() function. If not, we ignore it and don’t have to waste processor time sending the redundant data over the serial port. In my experience, this allows update rates of up to (possibly beyond) 1ms.

void oscEvent(OscMessage theOscMessage){
 if(theOscMessage.checkAddrPattern("/dac")==true){
   for(int i = 0; i < 8; i++){
     data[i] = theOscMessage.get(i).intValue();
     channelData[i] = data[i];
   }
   for(int i = 0; i < 8; i++){
     writeValue(i, data[i]);
   }
 }
}

Writing Serial Data to the Teensy

Serial teensy;
teensy = new Serial(this, Serial.list()[0], 19200);

The above lines are used in Processing to initialize a Serial object, allowing both read and write operations. The Serial.list()[0] argument indicates which actual serial port we want to write to. On my system, the Teensy always shows up as port 0 – this might be different on yours. Finally, the baud rate of 19200 is specified. Baud rate is the number of distinct signal events per second, and is a measure of data transfer speed.

Below is our writeValue() function, which was referenced above. The function is called repeatedly, once for each new sample to be written. First, we have to choose which of our two DAC chips should receive the data. Channels 0 – 3 go to chip A, 4 – 7 to chip B.

The MAX5250 is expecting a two byte word, which is assembled in the next section of code.

The SPI data expected by the MAX5250 DAC is as follows:
Screen shot 2010-08-27 at 10.48.25 AM

The first two bits select which of the four-per-chip channels to use, the second two bits allow us to write data with or without updating the actual voltage outputs, the next 10 bits are the actual data to be assigned, and the last two bits are unused. So, to write a data value of 512 to channel 3 and immediately output a voltage, we would send 1011001110110100.

As you can see, it’s a bit involved, and that’s why we want to avoid running all of this code unless the data has actually changed. We end up with three bytes to send to the Teensy 2.0, a one byte digit to indicate which DAC we want to write to, and the two additional SPI bytes. These are put into a buffer (really just an array) which is only transmitted when the buffer is full. This is to circumvent some timing weirdness in the USB to Serial conversion hardware.

void writeValue(int _channel, int _data){
 if(_channel > 3) { // assign one of two dac chips to respond
    dacChip = 1;
  } else {
    dacChip = 0;
  }

  /* bit shifting and masking to assemble proper list of bits for the DAC */
  _channel = _channel << 14;
  updateBits = 3 << 12;
  _channel = _channel | updateBits;
  _data = _data << 2;   spiWord = _channel | _data;
  binaryString = binary(spiWord, 16); // at this point, we've assembled our proper list of 16 bits
  outputData.add(byte(dacChip)); // so we'll throw them into an array, to facilitate transfer over serial
  outputData.add(byte(unbinary(binaryString.substring(0, 8))));
  outputData.add(byte(unbinary(binaryString.substring(8, 16))));
  if(outputData.size() >= 24){
    outputBytes = new byte[outputData.size()];
    for(int i = 0; i < outputData.size(); i++){
      outputBytes[i] = outputData.get(i);
    }
    teensy.write(outputBytes);
    dataIndex = 0;
    outputData = new ArrayList();
    previousUpdate = currentTime;
  }
}

Initializing SPI on the Teensy 2.0

Here’s an explanation of SPI from Wikipedia:

The SPI bus specifies four logic signals.

  • SCLK — Serial Clock (output from master)
  • MOSI/SIMO — Master Output, Slave Input (output from master)
  • MISO/SOMI — Master Input, Slave Output (output from slave)
  • SS — Slave Select (active low; output from master)

Essentially, the Master (Teensy 2.0 here) triggers the Slave chip by setting the SS pin low. Then the SCLK pin outputs a periodic clock pulse while the MOSI pin transmits the data (holding the SS pin low for the entire transfer). Here’s an image of the transmission from the MAX5250 datasheet – note that they use DIN (Data In) instead of MOSI, but it’s the same thing.

Screen shot 2010-08-27 at 10.24.55 AM

The first bit of code here is just a couple of statements to simplify our SPI communication. The DACs have a “Slave Select” pin, which allows them to either receive or ignore incoming data. This allows for easier wiring, you can connect all of the SPI lines to each chip, and just select which chip should respond at a given moment. Our DAC select byte (from above, in the writeValue() function) interfaces with the Slave Select code on the Teensy, and allows us to route data to the appropriate chip.Below, in the setup() function, we set the SS pins to output and set them both HIGH, so that no data is accidentally received by the DACs.

Finally, we call the setup_spi() function, found in Andrew Smallbone’s SPI library. These settings define how the Teensy should handle SPI, whether the DACs read the data on the rising or falling edge of the clock pulse, the SPI transmission rate as related to the Teensy clock, and a couple of other settings. You might notice that the serial interface is being initialized with a baud rate of 9600. The Teensy 2.0 actually ignores any baud rate argument and runs at full USB 2.0 speed.

#define SELECT_DAC_ONE digitalWrite(PORTB0, LOW);
#define DESELECT_DAC_ONE digitalWrite(PORTB0, HIGH);
#define SELECT_DAC_TWO digitalWrite(PORTD0, LOW);
#define DESELECT_DAC_TWO digitalWrite(PORTD0, HIGH);

void setup(){
  CPU_PRESCALE(CPU_4MHz);
  pinMode(PORTB0, OUTPUT);
  pinMode(PORTD0, OUTPUT);
  Serial.begin(9600);
  DESELECT_DAC_ONE;
  DESELECT_DAC_TWO;
  setup_spi(SPI_MODE_0, SPI_MSB, SPI_NO_INTERRUPT, SPI_MSTR_CLK2);
}

The last bit of code here reads incoming serial data, and immediately sends it out to the proper DAC. The serial buffering on the Teensy is a little bit different than the Arduino, in that it receives an entire USB packet at a time. The timing of the calls to Serial.read() can then be an issue. We want to make sure that we’re reading our three bytes in the proper order, and not getting out of phase with the host app, so we check that our first byte is either a 1 or a 0. Since the SPI interface packs data into the first and last bits of our data word (the second two bytes), a byte with the value of 1 or 0 will only appear as the first byte in the series. Timing is also important here, we need to introduce some brief delays so that we’re not reading or writing data too quickly.

void loop(){
  pollAndWrite();
}

void pollAndWrite(){
 data = false;
 while(!data){
  if(Serial.available()) { // look into the receive buffering - not receiving from Max properly
    firstByte = Serial.read();
    delayMicroseconds(100);
    if(firstByte == B00000000) {
      secondByte = Serial.read();
      delayMicroseconds(100);
      thirdByte = Serial.read();
      SELECT_DAC_ONE;
      send_spi(secondByte);
      send_spi(thirdByte);
      delayMicroseconds(10);
      DESELECT_DAC_ONE;
      data = true;
    }
      if(firstByte == B00000001){
        secondByte = Serial.read();
        delayMicroseconds(100);
        thirdByte = Serial.read();
        SELECT_DAC_TWO;
        send_spi(secondByte);
        send_spi(thirdByte);
        delayMicroseconds(10);
        DESELECT_DAC_TWO;
        data = true;
    }
  }
 }
}

So that’s the software side of the USB-Octomod. Although it’s fairly involved, there are only a few tricky spots, and the OSC interface greatly simplifies what the end-user actually has to think about during composition or performance.  Once the Processing and Teeny code is compiled and loaded, it becomes a plug-and-play device.

Lazers! Release on Stasisfield

Lazers! Cover Art

The Lazers! release, clocking in at around 100 minutes of music, is now available for free at Stasisfield.

Stasisfield curator John Kannenberg eloquently describes the recording:

Lazers! is a Milwaukee-based trio of composer/performers including David Collins, Steve Schlei, and Greg Surges. Their collective work focuses on custom-built instruments and indeterminate notation, incorporating situational, playful approaches to performance while generating exploratory sonic spaces. Electronic instrumentation comingles with structures and forms rooted in the acoustic realm of improvisation: look for beats elsewhere, this material is more about sound’s relationship to negative space than pattern and rhythm. A multitude of approaches and scenarios makes each track on their debut album a fresh experience, its own uniquely impermanent world.

Tracklist:

Improvisation (Silver) Part A

Mediation – David Collins, 2009

Improvisation (Silver) Part B

RadioGamelan – Greg Surges, 2010

Fission (Three Parts) – Greg Surges, 2009

Improvisation (Coumarin)

Fission (One Part) – Greg Surges, 2009

Improvisation (Stilbene)

Improvisation (Pink)

USB Control Voltage Interface – Progress So Far

With the arrival of the first prototype PCB today, I thought I would post a bit about the the design process thus far.

Moving from the very first Arduino-based prototype (breadboarded on a Radioshack experiment lab), I decided that a Teensy 2.0 would be a better solution. Cheaper, faster, breadboard capable, and a true serial peripheral, the chip works great and can be programmed using a slightly-modified version of the Arduino environment. You can also use the usual AVR toolchain if you prefer.

This is the original, very minimal prototype on a Radioshack experiment board.

This is the original, very minimal prototype on a Radioshack experiment board.

Setting up the Teensy 2.0 for breadboarding.

Setting up the Teensy 2.0 for breadboarding.

The next thing to do was to work on the software side of things. Originally, I developed a Max/MSP patch which directly translated integers into the proper binary string, as required by the MAX5250 DAC ICs. I wanted to make the project completely open source, so I decided to create a stand-alone program using Processing. The program essentially listens for specific OSC messages (which can be sent from any program or process the user chooses), and then translates those messages into 5250-compatible binary data. I think the solution works well, and removes any necessity for the user to own specific software.

Above, a Max/MSP patch transmits data over OSC, to the program below, which converts the received integers into binary data required by the DAC ICs.

Above, a Max/MSP patch transmits data over OSC, to the host program (below), which converts the received integers into binary data required by the DAC ICs.

After some final touches to the layout, (applying low-pass filters to smooth the output voltages, converting the unipolar output of the DAC to a bipolar voltage) I laid the circuit out in Eagle.

Here's the PCB layout as sent in for fabrication. There are a few small errors...

Here's the PCB layout as sent in for fabrication. There are a few small errors...

This is the silk-screened front of the PCB prototype.

This is the silk-screened front of the PCB prototype.

The back side, ground plane and all.

The back side, ground plane and all.

I’d say the board looks pretty good. I got it from BatchPCB.com, part of Sparkfun. Not exactly the cheapest or the fastest, but good for a one-off. (They gave me two…?)

Here’s the front again, with my hand for scale. The board is 2.9″ x 3.05″.

The board with a hand to show scale.

Hand-routed PCB

I ended up hand-routing the PCB. Here it is, as sent for fabrication.

I ended up hand-routing the PCB. Here it is, as sent for fabrication.

USB-CV Interface PCB Images

Here are some images of the pcb for my USB-CV device, made using the Eagle schematic/layout editor.

A closeup of the board, including part of the DAC IC.

A closeup of the board, including part of the DAC IC.

Part of the microcontroller, DAC, and opamp.

Part of the microcontroller, DAC, and opamp.

Voltage regulator, capacitor, and DAC.

Voltage regulator, capacitor, and DAC.

Finally, the full layout. Dimensions (in.) are 3.05 x 2.90.

Finally, the full layout. Dimensions (in.) are 3.05 x 2.90.