Using Arduino and Pd for musical live performance
Connecting an Arduino microcontroller to Pd is no big problem: You can just put Firmata on your Arduino and use the Pduino library for communication on the Pd side.
But sometimes Firmata is not the answer to every problem: You may want to turn on the built-in pull-up resistors for digital input pins, you may want to do some calculation on the Arduino before sending a message, or you may want to send a message from Arduino to Pd only on change.
In fact, I wanted to do all of the above, but not use Arduino for output. This led to the following electronics and code.
Connecting Inputs to Arduino
I have connected several push buttons to the Arduino, and one variable resistor. All push buttons are open an close on push. All connected devices:
- Three foot switches in a box.
- Three push buttons and a force-sensitive resistor mounted on a glove, worn on the right hand.
- A keypad consisting of 12 push buttons with a resistor matrix similar to this setup (in German). The main difference in my setup is the usage of values ten times as in the example, as the thin cables have a non-negligable resistance. This is mounted on my left forearm
Schematics
data:image/s3,"s3://crabby-images/eb823/eb823dc5007309dd2b0297a851ccd8318edb4b37" alt="circuit diagram"
The Arduino Shield
data:image/s3,"s3://crabby-images/b1419/b1419f46295ccb55f1471c5a7aa88d72f0ef5fd1" alt="The shield alone"
data:image/s3,"s3://crabby-images/0e0cf/0e0cf1c24657e923d0067b1f118bbabd0f9c21a7" alt="The shield with the PCB for keypad added"
data:image/s3,"s3://crabby-images/1a025/1a025cf8db7c6a34cc1ac9fd124cf1a8c816b110" alt="Fully mounted"
The Glove
data:image/s3,"s3://crabby-images/b9a14/b9a140c3d4d20968aca2766c0b5023a375badf6b" alt="Push buttons on the glove"
data:image/s3,"s3://crabby-images/ae5e0/ae5e0d7286587e1c85518a6e057364bff3761dcc" alt="Push buttons on the glove"
data:image/s3,"s3://crabby-images/9b52b/9b52b3ca646c463894ec3b392d839842a2d7f7cf" alt="Force sensitive resistor"
The Stompbox
data:image/s3,"s3://crabby-images/83f1b/83f1b642dbeba200e6a878d9c6dd8a25a9c5e75d" alt="Stompbox"
data:image/s3,"s3://crabby-images/16b38/16b380faeae87324c6703cb772ddf9d562b20785" alt="Stompbox"
The Keypad
data:image/s3,"s3://crabby-images/44171/441718202bc2ef98839e0a8e1bda4c67681563e6" alt="Keypad"
data:image/s3,"s3://crabby-images/e7f50/e7f501779f63b9cf753a6f67bf2c03f69a0bfb11" alt="Keypad"
Later Modifications Using Flex Sensors
data:image/s3,"s3://crabby-images/9a1ec/9a1ece4110c4af79c991fead9596ce10c5c24168" alt="Glove with flex sensors"
data:image/s3,"s3://crabby-images/ba6de/ba6de05d7c4a5df8949bc34aed1afe6a02099537" alt="Glove with flex sensors"
Message Passing between Arduino And Pd
I wanted to minimize the data sent from Arduino over USB to the host computer. I came up with the following "protocol", that uses 1 byte for a digital value and 2 bytes for analog values.
- Digital message
100ppppv
- Analog message
11pppvvv 0vvvvvvv
p
denotes bits used for encoding the pin number, v
bits used for values.
As the Arduino Duemilanove - the one that I use - has 14 digital pins, 4 bits (0-15) are necessary to encode the values, 6 analog ins can be encoded in 3 bits (0-7). Digital pins only read 0 or 1, so only one bit is needed for encoding the value, analog pins read values 0-1023, which corresponds to 10 bits.
If a byte starts with a value of 11, then this byte and the following one starting with 0 together are an analog message. If a byte starts with 100, then the message is a digital message.
On Pd's side, messages are parsed and then output in the same way as [pduino]
does it: (a|d) [pin number] [value]
. That design allows you use it as a drop in replacement.
I first tried to parse messages with bit shifting and [expr]
, but that was too CPU intensitive, so I have created my own external in C. That external does the bit shifting and then outputs messages as lists.
Pd Patch
I am using the setup mostly for live performances, and the demo patch included in the source code is the actual patch that I use. It also makes heavy use of my abstractions.