ThrowingBagels is a modernization of the LEDScape software package. ThrowingBagels a LED driver software package for the bagelbone BeagleBone Black, which drives LEDs from UDP packets you yeet at it over the network, or from your own custom software.
You can use ThrowingBagels to drive up to 32 channels of individual LED strips, up to 512 LEDs per channel. Any addressable LEDs that support the WS or SK control protocols should work (WS2812, SK6812, mainly). This should work on any BeagleBone Black, running Debian Buster.
Like we said, you can drive up to 32 channels of individual LED strips, up to 512 LEDs per channel. We've used versions of this software do build things like:
If you want LEDs to blink pretty, this gives you a powerful way to control them. All images above via Iontank's Facebook. See also the Iontank website.
curl https://raw.githubusercontent.com/iontank/ThrowingBagels/master/install.sh | bash
This script will:
- Clone ThrowingBagels into
/opt/ThrowingBagels
- Grant ownership of that directory to the current user (probably
debian
) - Build the code
- Provision the Beaglebone
- Create the leds.config file
- Change the hostname
- Update the
/boot/uEnv.txt
to disable HDMI and audio (we need those GPIO pins) - Install, but not enable, the systemd services for core applications
The primary use case is going to be running this as a UDP listener, but there are a few options for running services.
The first is bagel-launcher.service
, which simply installs the firmware and configures GPIO at boot. It's potentially useful for testing if you plan to interactively run your own software.
The second is multidemo.service
, which runs the multidemo application (see below).
The third is the main application, led-udp.service
, which runs the LED UDP RX program, and is the primary tool you'll likely use to control LEDs.
After install, all of these services will be installed, but none of them will be enabled.
For your usecase, you'll need to sudo systemctl enable $SERVICE
, where $SERVICE
is one of the above. You can then sudo systemctl start $SERVICE
.
The main driver of everything is the PRU Firmware, ledscape.asm
. This is what drives the GPIO channels to activate the LEDs.
For the purposes of testing, iterating, and diagnosing hardware issues, there are several included demo applications.
bin/smoke
(src/demos/smoke_test.c
): a simple smoke test application which sets the LEDs to a "soft-white" for ten seconds, and then back to black. It is the "hello world" of ThrowingBagels applications. (We don't go to full white, because that can overload many power supplies/distribution boards)bin/pru_status
(src/ledscape/pru_status.c
): a simple tool which checks the status of the PRU firmware.bin/multidemo
(src/demos/multidemo.c
): a complex and highly configurable suite of animations which run right on the BeagleBone, useful for demonstrating lighting effects without network connectivity. Great "pitch tool" to prove hardware concepts.
The primary application, and what drives the core use-case of this software stack, is the bin/led-udp-rx
tool (src/net/udp-rx.c
). This tool listens on a port (default: 9999
) for frame data. This allows a powerful "render machine" to run elsewhere on the network, and send frame data to an arbitrary number of ThrowingBagels instances. See LED UDP RX for more details.
Included is a launcher
script. This script configures the GPIO, deploys the firmware, and then, optionally, launches a requested binary. E.g., ./launcher bin/led-udp-rx
, would launch the LED UDP RX tool on port 9999, using leds.config
as the source for our led configuration.
Many of the tools, like led-udp-rx
and multidemo
, expect an LED configuration to be supplied. By default, they check leds.config
, but they both take a command line flag to change that. The file MUST be a single line, in this format:
number of channels,length of strips,number of bytes
e.g.:
32,100,4
: this would drive 32 channels of output, for strips 100 pixels long, and expects each LED to need 4 bytes (an RGBW LED). For LED UDP RX, this also sets the expected packet size- each packet should contain 50 pixels for all 32 channels (see the UDP RX Readme for more details).
scripts/provision
will also prompt you for this information during the install.
To build your own application running on the BeagleBone, you must include two files. First, you need to include ledstrips.h
into your C code, and you must include a LUT: either gamma8.h
(recommended), linear.h
, or your own custom LUT.
From there, you need to flatten your frames into a stripwise structure, e.g. you should be able to access individual pixels using fb[(y*w)+x]
(that access the pixel at the given x/y coordinate in a 1D version of your framebuffer). See modes.h for examples.
The key methods you'll use are:
strip_config* leds_config(char const * filename)
: load a config file and return the strip configint leds_init(strip_config *cfg)
: give a strip config, this connects to the PRU shared memory. Returns1
on success,0
on failurevoid leds_draw(strip_config * cfg, const void * const frame)
: drawsframe
data to the LEDs. This needs to follow the specification above, and must bewidth*height*4
bytes. This will flush the contents of the frame to the LEDs for displayvoid leds_wait(strip_config * cfg)
: wait for the LEDs to draw. You never need to call this, as the PRU is almost certainly flushing the data faster than your program is generating it, but just in casevoid leds_close(strip_config * cfg)
: releases our hold on the PRU shared memory.
See ledstrips.h for more details.
Running the software is useless if you can't plug LEDs into it. A reference design for a board is included. It is based on designs we've used in our installations, but is strictly a reference design.