Simple DIY Electronic Music Projects<p><strong>Duppa I2C MIDI Controller – Part 1</strong></p><p>I’ve had my eye on <a href="https://www.duppa.net/" rel="nofollow noopener noreferrer" target="_blank">Duppa’s</a> rather neat looking I2C LED rings and rotary encoders for some time and finally pushed “go” on getting a couple. I want to use them as the basis for a MIDI controller but this post is just my “getting to know them” post.</p><p>A follow-up will build on this knowledge to get them going with MIDI.</p><p><em><strong>Warning!</strong> I strongly recommend using old or second hand equipment for your experiments. I am not responsible for any damage to expensive instruments!</em></p><p>If you are new to Arduino, see the <a href="https://diyelectromusic.wordpress.com/getting-started/" rel="nofollow noopener noreferrer" target="_blank">Getting Started</a> pages.</p><p><strong>Parts list</strong></p><ul><li>Arduino Uno/Nano</li><li>DuPPa small RGB LED Ring and mini I2C Encoder.</li><li>Bespoke hook-up wires (available from duppa.net).</li><li>Breadboard and jumper wires</li></ul><p><strong>Duppa I2C Devices</strong></p><p>There are several interesting devices, two variants of LED ring, and two variants of I2C connected rotary encoders. They are all pretty interesting devices, and the cascading nature of, especially the I2C encoders, has massive potential for a range of interesting MIDI controllers!</p><p>I have the following devices:</p><ul><li><a href="https://www.duppa.net/shop/i2c-encoder-mini/" rel="nofollow noopener noreferrer" target="_blank">I2CEncoder Mini</a></li><li><a href="https://www.duppa.net/shop/rgb-led-ring-small/" rel="nofollow noopener noreferrer" target="_blank">RGB LED Ring Small</a></li></ul><p>A couple of points to note about these:</p><ul><li>The LED ring has jumpers for I2C address and optional I2C pull-ups.</li><li>The encoder has a software-configurable I2C address and optional I2C pullups.</li><li>Both have a supporting library and code on GitHub here: <a href="https://github.com/Fattoresaimon/ArduinoDuPPaLib" rel="nofollow noopener noreferrer" target="_blank">https://github.com/Fattoresaimon/ArduinoDuPPaLib</a></li><li>Both have hardware designs and additional information on GitHub too:<ul><li><a href="https://github.com/Fattoresaimon/I2CEncoderMini" rel="nofollow noopener noreferrer" target="_blank">https://github.com/Fattoresaimon/I2CEncoderMini</a></li><li><a href="https://github.com/Fattoresaimon/RGB_LED_Ring_Small" rel="nofollow noopener noreferrer" target="_blank">https://github.com/Fattoresaimon/RGB_LED_Ring_Small</a></li></ul></li></ul><p><strong>Duppa LED Rings</strong></p><strong>LED Ring Small</strong><strong>LED Ring</strong>24 RGB LEDs48 RGB LEDsIS31FL3746A controllerIS31FL3745 controllerCascade up to 15 ringsCascade up to 16 rings<p>Both LED rings support the following (from the product pages):</p><ul><li>Individual 256 PWM control steps</li><li>Individual 256 DC current steps</li><li>Global 256 step current setting</li><li>29kHz PWM frequency</li><li>Individual open and short error detect function</li><li>1MHz I2C-compatible interface</li><li>Supply voltage range: 2.7V to 5.5V</li></ul><p>And they share a cable interface using a Molex “Picoblade” connector at 1.25mm pitch, with the following pinout:</p>VCCLED supply voltage2.7V – 5.5VGNDGround0VVIOLogic supply voltage2.7 – 5.5VSDAI2C dataSame as VIOSCLI2C clockSame as VIO<p>Note, at first glance, especially as these are I2C devices, it is easy to imagine that these are compatible with standards such as Qwicc, Stemma, Grove, and so on. They are not (<a href="https://emalliab.wordpress.com/2025/02/11/stemma-grove-qt-gravity-duppa-i2c/" rel="nofollow noopener noreferrer" target="_blank">more here</a>).</p><p>These are designed so that the LEDs can be powered independently from the logic to allow for higher current if required. This includes using 5V for LEDs and 3V3 for the logic. Apparently both rings state a “all LEDs on” current of 450mA. I don’t know if this is a typo…</p><p>The connectors are paired so it is possible to have an “in” and “out” connection allowing for up to 15 or 16 rings to be cascaded, providing different I2C addresses are set via the configuration jumpers. The address range is 0x60 to 0x7E (in steps of two).</p><p>Naturally using more rings will definitely require an independent LED power supply that can provide the maximum required current.</p><p><strong>Duppa I2C Encoders</strong></p><p>There are two variants of I2C encoder (actually there are also some older versions of them too, which are not compatible, but I’m ignoring those).</p><strong>I2CEncoder Mini</strong><strong>I2C Encoder</strong>Programmable I2C addressSolder jumper I2C addressATtiny 402PIC16F18345Cascade in one dimensionCascade in two-dimensional matrixRGB LED support3 GPIO pins<p>Some common features:</p><ul><li>3V3 to 5V operation.</li><li>Available with 2.54mm pin headers or JST-XH sockets.</li><li>400 kHz I2C.</li><li>Interrupt line.</li><li>Optional I2C pull-ups via solder jumper.</li></ul><p>Both share a common 2.54mm pitch connector and has options for pin headers or JST-XH sockets fitted. The pinout is slightly different to the LED rings, as follows:</p>GNDGroundVCC3V3 to 5V powerSDAI2C DataSCLI2C ClockINTOpen drain interrupt signal.<p>Once again these might superficially appear compatible with the likes of Grove and other I2C common options, but they are not.</p><p>Also, pretty critically, it should be noted that GND and VCC are the opposite way round to the LED rings, and that SDA/SCL are in different places in the five pins.</p><p>The I2C Address for the Mini encoder has to be set in software. The default is 0x20, but it would appear that any of the 128 addresses (apart from 0) could be used. A utility sketch is <a href="https://github.com/Fattoresaimon/ArduinoDuPPaLib/blob/master/examples/I2CEncoderMini/Change_Address/Change_Address.ino" rel="nofollow noopener noreferrer" target="_blank">provided on GitHub</a> and as part of the support library (see below), to set the address.</p><p>Note: the full LED ring has 7 address jumpers which will set the 7-bit address in binary directly in hardware.</p><p>There are datasheets for both encoders in the respective GitHub repositories. <a href="https://github.com/Fattoresaimon/I2CEncoderMini/blob/master/I2CEncoderMini_v1.2.pdf" rel="nofollow noopener noreferrer" target="_blank">Here is the one for the I2CEncoder Mini</a>.</p><p><strong>Getting Started</strong></p><p>My plan is to use an LED ring and encoder as a pair, on the I2C bus as a MIDI controller. This means deciding on I2C addresses, pull-up configuration, and connections.</p><p>All Duppa devices have options for buying compatible cables, so I would advise getting a range for experimentation. My first task was to crimp some jumper header pins onto each of the cables.</p><p>For my initial experiments I’ve gone with the following configuration</p><ul><li>Enable I2C pull-ups on the LED ring.</li><li>Use LED ring base address of 0x60 (solder S1, S5).</li><li>Use Encoder default base address of 0x20.</li></ul><p>This requires the I2C pull-up, S1, and S5 solder bridges making on the LED ring, but means I can use the encoder “as is” (for now).</p><p>The following shows the connections to an Arduino Nano. </p><p>Note that this is powering the LED ring directly from the Nano’s 5V, which itself is powered from the USB socket. Most standard USB sockets would be good for around 500mA I believe, so this should be ok, especially if all LEDs are not on at the same time at full brightness.</p><p>Other things to note:</p><ul><li>For the 5-wire links to the devices, I’ve used the colours that came with the wires I’d ordered from Duppa.net.</li><li>Notice how 5V/GND are swapped for the two devices.</li><li>I’ve used 5V for both the LED VCC and logic VIO for the LED ring.</li><li>In both cases SDA/SCL are cascaded up to the Nanos A4 (SDA), A5 (SCL).</li><li>The encoders interrupt line is connected to A3.</li></ul><p><strong>A note on pull-ups</strong></p><p>As already mentioned, I’ve enabled hardware pull-up resistors on the I2C bus by enalbing the solder bridge on the LED ring to connect both on-board I2C 4K7 pull-ups to VCC. Another option would have been to add external 4K7 resistors on the solderless breadboard between both SDA and SCL and VCC (5V).</p><p>Another option would have been to solder bridge the I2C pull-ups on the encoder instead.</p><p>Generally speaking I2C needs pull-ups somewhere and there are several approaches:</p><ul><li>Use a microcontroller with hardware pull-ups on SDA and SCL (internal pull-ups aren’t usually “strong” enough for I2C as a rule);</li><li>Add pull-ups on one of the devices connected to the bus;</li><li>Use external pull-ups.</li></ul><p>There are pros/cons of each. The Arduino doesn’t include additional pull-ups on I2C, but the Raspberry Pi does. Adafruit devices often include an option for enabling pull-ups on each device. It might be possible to use the internal pull-ups on a microcontroller, but they are often considered to be “weak” pull-ups (on an ATMega328 internal pull-ups around ~20-50k) and may not be suitable for the higher speeds of the I2C bus.</p><p>Only one set of pull-ups is typically required on the bus. Enabling more may work, but essentially puts the pull-ups in parallel with each other, which lowers the resistance used. I2C seems pretty tolerant of a wide range of pull-up values, so it might be fine for many purposes, but eventually it could lead to current draw issues one presumes.</p><p>There is also the issue of what a pull-up should pull up to. If a device is powered using 5V, but the I2C bus is running at 3V3 because it is interfacing to a 3V3 logic microcontroller, then pull-ups need to be 3V3 too.</p><p>Some references for pull-up resistors:</p><ul><li><a href="https://learn.sparkfun.com/tutorials/pull-up-resistors/" rel="nofollow noopener noreferrer" target="_blank">https://learn.sparkfun.com/tutorials/pull-up-resistors/</a></li><li><a href="https://learn.adafruit.com/working-with-i2c-devices/pull-up-resistors" rel="nofollow noopener noreferrer" target="_blank">https://learn.adafruit.com/working-with-i2c-devices/pull-up-resistors</a></li><li><a href="https://support.arduino.cc/hc/en-us/articles/11153357842588-I2C-and-pull-up-resistors" rel="nofollow noopener noreferrer" target="_blank">https://support.arduino.cc/hc/en-us/articles/11153357842588-I2C-and-pull-up-resistors</a></li></ul><p>The interrupt signal for the I2CEncoder should also include a pull-up resistor, but in this case, as it is not part of the high-speed I2C bus, the internal resistor of either the microcontroller or encoder should suffice.</p><p>There is a configuration option for the encoder to tell it to enable the pull-up on the interrupt line, so this is used.</p><p><strong>The Code</strong></p><p>There is an Arduino library available for all of the Duppa products here: <a href="https://github.com/Fattoresaimon/ArduinoDuPPaLib/" rel="nofollow noopener noreferrer" target="_blank">https://github.com/Fattoresaimon/ArduinoDuPPaLib/</a></p><p>This has to be downloaded as a ZIP file of the whole repository and then loaded into the Arduino environment using the “Include Library”->”Add ZIP Library” option.</p><p>To verify what I2C addresses are being used, a simple I2C scanner sketch can be used: <a href="https://playground.arduino.cc/Main/I2cScanner/" rel="nofollow noopener noreferrer" target="_blank">https://playground.arduino.cc/Main/I2cScanner/</a></p><p>In my case this returns:</p><pre>I2C Scanner<br>Scanning...<br>I2C device found at address 0x20 !<br>I2C device found at address 0x60 !<br>done</pre><p>As already mentioned if the I2C address of the I2CEncoder Mini needs to be changed there is a sketch in the DuPPa library examples to do this:</p><ul><li>Examples -> DuPPa Library -> I2CEncoderMini -> Change_Address.</li></ul><p>It is worth ensuring the provided examples work before going further.</p><p><strong>LED Ring Demo</strong></p><ul><li>Examples -> DuPPa Library -> RGB LED Ring Small -> LEDRingSmall_Demo</li></ul><p>The I2C address can be set at the start as follows:</p><pre>LEDRingSmall LEDRingSmall(ISSI3746_SJ1 | ISSI3746_SJ5);</pre><p>I also limited the “global current” by changing 0xFF (255) in two places near the end of the loop() to 0x20 (32):</p><pre>for (i = 0x20; i > 0; i--) { // Was 0xFF<br> LEDRingSmall.LEDRingSmall_GlobalCurrent(i);<br> delay(20);<br>}<br>LEDRingSmall.LEDRingSmall_ClearAll();<br>LEDRingSmall.LEDRingSmall_GlobalCurrent(0x20); // Was 0xFF</pre><p>There are a few other calls to LEDRingSmall_GlobalCurrent() which the library API notes sets the Global Current Control Register in the IS31FL3746A according to Table 10 in the datasheet.</p><p>I must admit it isn’t totally clear to me exactly how this works, but various descriptions describe a limiting function on the global current used in 256 steps with 0 being the lowest and 255 (0xFF) being the highest/brightest. The datasheet says that the default is 0, so presumably if this call isn’t used then there will be no output. One section of the demo turns on all LEDs and steps through the global current values from 255 down to 0. I changed this to 32 (0x20) down to 0.</p><p>In fact this is a common feature of the library, it appears to be a fairly shallow shim over the direct registers of the IS31FL3746A, so some knowledge of the datasheet is required to use it properly. Although I suspect that for many cases, just starting with the default demo program and fiddling about will probably go quite a long way.</p><p>The main LED functions, once the setup is complete, take a LED number and will either set a RGB colour or can set the individual colour levels of red, green or blue from 0 to 255.</p><pre>led = 0..23<br>rgb = 0x000000 .. 0xFFFFFF<br>LEDRingSmall_Set_RGB(led, rgb);<br><br>led = 0..23<br>level = 0 .. 255<br>LEDRingSmall_Set_RED(led, level);<br>LEDRingSmall_Set_GREEN(led, level);<br>LEDRingSmall_Set_BLUE(led, level);</pre><p><strong>I2C Encoder Demo</strong></p><ul><li>Examples -> DuPPa Library -> I2CEncoderMini -> Basic_with_Callbacks</li></ul><p>The I2C address and Interrupt pin are required:</p><pre>const int IntPin = A3;<br>i2cEncoderMiniLib Encoder(0x20);</pre><p>The demo code uses callbacks from the library to perform various actions depending on the state of the encoder. The following events are tracked (and reported via the serial console):</p><pre>onIncrement<br>onDecrement<br>onMax<br>onMin<br>onButtonPush<br>onButtonRelease<br>onButtonDoublePush<br>onButtonLongPush</pre><p>The interrupt pin is enabled using autoconfigInterrupt() and also relies on the internal pull-up being enabled on the encoder. This is part of the configuration provided on initialisation.</p><pre>i2cEncoderMiniLib Encoder(0x20);<br><br>Encoder.reset();<br>Encoder.begin( i2cEncoderMiniLib::WRAP_DISABLE<br> | i2cEncoderMiniLib::DIRE_LEFT<br> | i2cEncoderMiniLib::IPUP_ENABLE<br> | i2cEncoderMiniLib::RMOD_X1<br>);</pre><p>Whilst the interrupt pin is enabled at the encoder end, it doesn’t actually cause an interrupt on the Arduino. The main loop just polls the relevant INPUT pin and uses it to decide if the encoder needs checking:</p><pre>void loop() {<br> if (digitalRead(IntPin) == LOW) {<br> /* Check the status of the encoder and call the callback */<br> Encoder.updateStatus();<br> }<br>}</pre><p>I’m guessing this is better than continually scanning the I2C bus to see if there is anything to do, assuming there is a spare GPIO pin to connect up.</p><p>Some other points of note for use of the library:</p><ul><li>It is possible to enable/disable some of the interrupt sources using writeInterruptConfig(), but it can also be automatically configured by calling autoconfigInterrupt() once all the callbacks have been specified for just the sources that match the callbacks.</li><li>There are three types of encoder supported, representing by the X1, X2, X4 definitions.</li><li>There is a method for ChangeI2CAddress() but this should not be used in general purpose code. It only really makes sense in the utility sketch already mentioned for one-time setting of the I2C address of the encoder.</li><li>There is a block of EEPROM that can be accessed using readEEPROM()/writeEEPROM() which might come in handy.</li></ul><p><strong>Encoder driven LED Ring</strong></p><p>Combining elements of both demos here is a short sketch that uses the encoder to rotate an LED pattern around the ring.</p><pre>#include <Wire.h><br>#include <i2cEncoderMiniLib.h><br>#include "LEDRingSmall.h"<br><br>const int IntPin = A3;<br>i2cEncoderMiniLib Encoder(0x20);<br>LEDRingSmall LEDRingSmall(ISSI3746_SJ1 | ISSI3746_SJ5);<br><br>#define MAX_RGB 0x7F<br><br>int tim,last;<br>int dir;<br>int r, g, b, rg, br, bg;<br>#define INCR(x) (x)+=dir; if ((x)>=24) {(x)=0;} else if ((x)<0) {(x)=23;}<br><br>void encoder_increment(i2cEncoderMiniLib* obj) {<br> dir = 1;<br> tim ++;<br>}<br><br>void encoder_decrement(i2cEncoderMiniLib* obj) {<br> dir = -1;<br> tim--;<br>}<br><br>void setup() {<br> pinMode(IntPin, INPUT);<br> dir = 0;<br> tim = 0;<br> last = 1;<br><br> Wire.begin();<br> Wire.setClock(400000);<br><br> LEDRingSmall.LEDRingSmall_Reset();<br> delay(20);<br><br> LEDRingSmall.LEDRingSmall_Configuration(0x01); //Normal operation<br> LEDRingSmall.LEDRingSmall_PWMFrequencyEnable(1);<br> LEDRingSmall.LEDRingSmall_SpreadSpectrum(0b0010110);<br> LEDRingSmall.LEDRingSmall_GlobalCurrent(0x10);<br> LEDRingSmall.LEDRingSmall_SetScaling(0xFF);<br> LEDRingSmall.LEDRingSmall_PWM_MODE();<br><br> Encoder.reset();<br> Encoder.begin(i2cEncoderMiniLib::WRAP_DISABLE<br> | i2cEncoderMiniLib::DIRE_LEFT<br> | i2cEncoderMiniLib::IPUP_ENABLE<br> | i2cEncoderMiniLib::RMOD_X1 );<br><br> Encoder.writeCounter((int32_t) 0); /* Reset the counter value */<br> Encoder.writeStep((int32_t) 1); /* Set the step to 1*/<br><br> Encoder.onIncrement = encoder_increment;<br> Encoder.onDecrement = encoder_decrement;<br><br> Encoder.autoconfigInterrupt();<br>}<br><br>void loop() {<br> dir = 0;<br> if (digitalRead(IntPin) == LOW) {<br> Encoder.updateStatus();<br> }<br><br> if (tim != last) {<br><br> if ((tim % 1) == 0) {<br> LEDRingSmall.LEDRingSmall_Set_RED(r, 0);<br> INCR(r);<br> LEDRingSmall.LEDRingSmall_Set_RED(r, MAX_RGB);<br> }<br><br> if ((tim % 2) == 0) {<br> LEDRingSmall.LEDRingSmall_Set_GREEN(g, 0);<br> INCR(g);<br> LEDRingSmall.LEDRingSmall_Set_GREEN(g, MAX_RGB);<br> }<br><br> if ((tim % 3) == 0) {<br> LEDRingSmall.LEDRingSmall_Set_BLUE(b, 0);<br> INCR(b);<br> LEDRingSmall.LEDRingSmall_Set_BLUE(b, MAX_RGB);<br> }<br><br> if ((tim % 4) == 0) {<br> LEDRingSmall.LEDRingSmall_Set_BLUE(br, 0);<br> LEDRingSmall.LEDRingSmall_Set_RED(br, 0);<br> INCR(br);<br> LEDRingSmall.LEDRingSmall_Set_BLUE(br, MAX_RGB);<br> LEDRingSmall.LEDRingSmall_Set_RED(br, MAX_RGB);<br> }<br><br> if ((tim % 5) == 0) {<br> LEDRingSmall.LEDRingSmall_Set_BLUE(bg, 0);<br> LEDRingSmall.LEDRingSmall_Set_GREEN(bg, 0);<br> INCR(bg);<br> LEDRingSmall.LEDRingSmall_Set_BLUE(bg, MAX_RGB);<br> LEDRingSmall.LEDRingSmall_Set_GREEN(bg, MAX_RGB);<br> }<br><br> if ((tim % 6) == 0) {<br> LEDRingSmall.LEDRingSmall_Set_RED(rg, 0);<br> LEDRingSmall.LEDRingSmall_Set_GREEN(rg, 0);<br> INCR(rg);<br> LEDRingSmall.LEDRingSmall_Set_RED(rg, MAX_RGB);<br> LEDRingSmall.LEDRingSmall_Set_GREEN(rg, MAX_RGB);<br> }<br> }<br> last = tim;<br>}</pre><p>This only uses the increment/decrement callbacks to set the direction (“dir”) and update the time (“tim”) that determines which direction the ring is moving.</p><p>It isn’t perfectly reversible – I liked the idea of winding the ring forward and back again – but it is pretty close. I suspect there is an off-by-one error somewhere that means it can get out of sync.</p><p>But it is plenty enough to show how it all works.</p><p><strong>Closing Thoughts</strong></p><p>These are great little boards. Having an I2C encoded in my case might be a bit over the top, but they are pretty easy to use.</p><p>The LED ring is awesome 🙂</p><p>Kevin</p><p><a rel="nofollow noopener noreferrer" class="hashtag u-tag u-category" href="https://diyelectromusic.com/tag/duppa/" target="_blank">#duppa</a> <a rel="nofollow noopener noreferrer" class="hashtag u-tag u-category" href="https://diyelectromusic.com/tag/i2c/" target="_blank">#i2c</a> <a rel="nofollow noopener noreferrer" class="hashtag u-tag u-category" href="https://diyelectromusic.com/tag/is31fl3746a/" target="_blank">#is31fl3746a</a> <a rel="nofollow noopener noreferrer" class="hashtag u-tag u-category" href="https://diyelectromusic.com/tag/rgb-led/" target="_blank">#rgbLed</a> <a rel="nofollow noopener noreferrer" class="hashtag u-tag u-category" href="https://diyelectromusic.com/tag/rotary-encoder/" target="_blank">#rotaryEncoder</a></p>