-
Notifications
You must be signed in to change notification settings - Fork 131
WLED Programming Notes
This version of WLED contains sound reactive routines, which are prefaced with a * FX_NAME
for volume reactive routines and ** FX_NAME
for FFT routines. Each is controllable by the intensity and/or speed sliders. Some routines also include 3 dedicated FFT control sliders.
Contents:
- HTTP API Links, Updating FX.h, Updating FX.cpp
- WLED Support Functions
- Important WLED variables, Important Structures
- Delays in WLED, Displaying LEDs, FastLED, 2D Functionality
- Variable Length Arrays, Using CRGB Color Space, Using CHSV Colour Space
- Non-Dynamically Created Variable Arrays
- Sound Reactive EEPROM Layout
Caveat: Some information on this page is in our 'dev' branch only and not yet ready for prime time.
I hope to work on an Android application in the future, so I'm keeping notes on the HTTP API.
- https://tynick.com/blog/01-28-2020/controlling-wled-with-raspberry-pi-using-the-wled-api/
- https://github.com/Aircoookie/WLED/wiki/HTTP-request-API
- Line 105 – Update the mode count.
- Line 250 - Added extern variables around line 250 of FX.h. These are defined in audio_reactive.h.
- Line 214 – Update the defines.
- Line 456 – Update the _mode.
- Line 697 – Update the mode definition and don’t forget , and ;
- Line 816 – Update the JSON entry. Can add an extra entry. No Spaces, 10 per line!!
Append the new function(s) and use current functions as templates. Cannot add any functions without accompanying index entry.
-
fade_out(uint8_t rate);
- This is CRITICAL!!! -
blur(uint8_t blur_amount);
- Can be useful. -
color_wheel(uint8_t index);
- This has history with NeoPixel library. I use palettes instead.
-
SEGLEN
// uint16_t - Segment length. -
_segment_index
// uint8_t - Current segment being displayed. -
SEGMENT.length
// uint16_t - Segment length (but not for ESP8266 :^/ ) -
SEGMENT.intensity
// uint8_t - You can use this from the slider. -
SEGMENT.speed
// uint8_t - You can use this from the slider. -
SEGMENT.palette
// uint8_t - Current palette. Otherwise SEGCOLOR(0) and SEGCOLOR(1). -
SEGMENT.mode
// uint8_t - Current mode. -
now
// uint32_t – Millis counter. -
SEGENV.call
// uint32_t – Counter each time a routine is called. Can be used for 'setup'. -
SEGENV.next_time
// uint32_t – Millis counter. -
SEGENV.step
// uint32_t - Counter each time a routine is called. -
SEGENV.aux0
// uint16_t - Available for use as a SEGMENT specific persistent variable. -
SEGENV.aux1
// uint16_t - Available for use as a SEGMENT specific persistent variable.
- FX.h:267 - Segment aka SEGMENT
- FX.h:318 - Segment runtime.value> aka SEGENV
- FX.h:66 - Other stuff
You DO NOT use delay statements here, except to keep the watchdog happy. Here is the awesome FastLED method of timing/scheduling:
EVERY_N_MILLISECONDS_I(pixTimer, SEGMENT.speed) {
pixTimer.setPeriod(256 - SEGMENT.speed);
// Put your display code here.
}
Here's the standard blink without delay version:
long lastTime = 0;
int delayMs = 2000; // We want to do something every 2 seconds.
void userLoop()
{
if (millis()-lastTime > delayMs)
{
lastTime = millis();
//do something you want to do every 2 seconds
}
}
We were used to FastLED.show(). Well, no longer, and this has some disadvantages.
CRGB myCol = ColorFromPalette(currentPalette, index, brightness, LINEARBLEND);
setPixelColor(myLED, myCol.red, myCol.green, myCol.blue);
The next line supports SEGCOLOR(0) and SEGCOLOR(1) if no palette (i.e. default) is selected:
setPixelColor(i, color_blend(SEGCOLOR(1), color_from_palette(index, false, PALETTE_SOLID_WRAP, 0), pixBri));
where, i
is the location, index
is the color and pixBri
is the brightness.
Now, onto the disadvantage. . The problem is that you can perform a getPixelColor
and move it to another LED with setPixelColor
. After moving to about 10 pixels, the LED is now black. This is because of the built-in scaling and addressing the memory that's used for DMA transfer. Unfortunately, we don't get a nice lossless leds[location]
as we do with FastLED.
WLED uses the NeoPixelBus library to drive the LED's directly, however, a significant amount of FastLED functionality has been enabled in WLED. Things included:
- FastLED Math and trigonometry
- Noise
- Palettes
- Palette transitioning (use your own palette variables though)
- EVERY_N_MILLIS
What is NOT included:
FastLED.show()
- FastLED pixel setup
- FastLED power management
- Working directly with the
leds[x]
array -
fill_rainbow
and related routines that directly affect the array -
fade
/nscale
, you need to use the WLED equivalent
- An
XY()
function which returns the index number of the X-Y coordinate on the matrix has been added.- The origin (x=0,y=0) is at the top left of the display, with X axis extending to the right, Y axis toward the bottom
- The LED order starts from 0 at the top left, and ascends to the right on the top row
- 2D addressing is leds[XY(column, row)]
- A serpentine/zig-zag setting has been added to 'LED Settings'.
- With serpentine mode enabled, the led index on every odd row (top row is row 0) is ascending from right to left, and on even rows is ascending from left to right
- The opposite of serpentine is "progressive", where all rows are oriented in ascending order from left to right
Don't enable serpentine setting if your pixels are
laid out all running the same way, like this:
0 > 1 > 2 > 3 > 4
|
.----<----<----<----'
|
5 > 6 > 7 > 8 > 9
|
.----<----<----<----'
|
10 > 11 > 12 > 13 > 14
|
.----<----<----<----'
|
15 > 16 > 17 > 18 > 19
Enable serpentine setting if your pixels are
laid out back-and-forth, like this:
0 > 1 > 2 > 3 > 4
|
|
9 < 8 < 7 < 6 < 5
|
|
10 > 11 > 12 > 13 > 14
|
|
19 < 18 < 17 < 16 < 15
-
matrixWidth
andmatrixHeight
are new 'LED Settings'.-
matrixWidth
is the number of pixels per row -
matrixHeight
is the number of pixels per column
-
- Some existing animations (as of Jan 2021) were coded for different orientations and need to be redone to match the newly agreed to orientation described above
- Not yet supported: rotation, or any mapping other than rectangular progressive/serpentine
- Ideally the
XY()
function will return the correct led index even with a rotated display, andmatrixWidth
/matrixHeight
will refer to the rotated display's dimensions. - It's likely that there will at least a minor change in animations required to support a non-rectangular or non-contiguous mapping
- Ideally the
These will be used so that we can address leds[x] in our routines with the known and lossless FastLED method rather than the lossy NeoPixelBus method. We'll also be using these for ancillary functions, i.e. heat[x] as used in fire2012.
Since you cannot define variable arrays in C, we need other methods to do so for our functions.
The WLED method has been to malloc() some memory as follows:
if (!SEGENV.allocateData(SEGLEN)) return mode_static();
byte *heat = SEGENV.data;
heat[value] = 25;
For the 2D and FastLED data array functionality, the developers of this fork are not comfortable with the malloc()
method of memory allocation and have decided to create large fixed arrays instead and to create pointers for use with variable-length arrays.
Already declared in FX.cpp:
uint32_t ledData[MAX_LEDS]; // Currently 1500 as defined in const.h. Used for conversion from NeoPixelBus to FastLED. RGB or RGBW.
uint32_t dataStore[4096]; // For ancillary functions. Can use any data type.
CRGB *leds = (CRGB *)ledData; // Define the pointer and override the default data type.
leds[0] = CRGB::Red;
leds[1] = ColorFromPalette(currentPalette, index, bright, LINEARBLEND);
First, define a function and add to FX.h
void WS2812FX::setPixels(CRGB* leds) {
for (int i=0; i<SEGLEN; i++) {
setPixelColor(i, leds[i].red, leds[i].green, leds[i].blue);
}
}
You can now use FastLED address methods, AND it works with segments.
Then, you can call this function at the end of your routine as follows:
setPixels(leds);
return FRAMETIME;
} // End of your routine.
CHSV *leds = (CHSV *)ledData;
leds[i] = CHSV(25, 255,255);
The leds[]
array is already defined in the CHSV color space.
CRGB color;
for (int i=0; i<SEGLEN; i++) {
color = leds[i];
setPixelColor(i, color.red, color.green, color.blue);
}
As used in one of the FFT animations (** Freqmatrix) in FX.cpp around line 4079.
uint32_t *leds = ledData;
CHSV c;
c = CHSV(20, 255, 255);
leds[i] = (c.h << 16) + (c.s << 8) + (c.v );
for (int i=0; i<SEGLEN; i++) {
c.h = (leds[i] >> 16) & 0xFF;
c.s = (leds[i] >> 8) &0xFF;
c.v = leds[i] & 0xFF;
color = c;
setPixelColor(i, color.red, color.green, color.blue);
}
We'll use that large dataStore array that was defined globally in FX.cpp. Although it was defined as a unint32_t
, you can still use smaller dynamic arrays and override the data type using pointers with a function:
uint8_t *myArray = (uint8_t *)dataStore; // Just make sure you don't go over.
You can now use it as a 1D or quasi 2D array, i.e.
myArray[5] = 4; // 1D array
myArray[5*matrixWidth + 4] = 10; // 2D array
We've expanded EEPSIZE
in const.h to 4095 for ESP32 and 3300 for ESP8266 due to our additional requirements. We started saving values at 3072 defined as EEP_AUDIO
. Although we'd like to apply SEGMENT specific settings, we may have some challenges with the FFT sliders. We're not sure at this point.
Byte | Data |
---|---|
EEP_AUDIO | Sound Squelch Setting |
EEP_AUDIO+1 | Audio Sync Port |
EEP_AUDIO+2 | Audio Sync Port |
EEP_AUDIO+3 | Audio Sync Enabled |
EEP_AUDIO+4 | FFT1 Slider Value |
EEP_AUDIO+5 | FFT2 Slider Value |
EEP_AUDIO+6 | FFT3 Slider Value |
EEP_AUDIO+7 | Begin 2D Matrix Values |
EEP_AUDIO+11 | End 2D Matrix Values |
EEP_AUDIO+12 | Input Gain Setting |
How do we store FFT slider values in EEPROM for WLED presets? WLED Presets are 20-byte blocks (slots) stored in EEPROM. There is space reserved in EEPROM for 25 slots from 400-899. Currently, 18 of the 20 bytes are being used by WLED. This presents a problem for us since, at the time of writing, we have 3 bytes that we need to store for our FFT sliders. We didn't want to attempt to rewrite the entire WLED preset protocol as that would surely introduce unnecessary headaches.
To solve this problem, we reserved 25 5-byte blocks (slots) in EEPROM from 3175-3299. With the space in EEPROM allocated, we can now save/retrieve FFT slider data to/from WLED presets.
Byte | Data |
---|---|
0 | FFT1 Slider Value |
1 | FFT2 Slider Value |
2 | FFT3 Slider Value |
3 | ZERO (Reserved) |
4 | ZERO (Reserved) |
WLED limits the frame rate, and it's apparently because the LED's start flashing if the frame rate is too high. The standard for maintaining a consistent frame rate when writing any animations is to make sure you add to the end of your animation:
return FRAMETIME;
If you wish to increase the frame rate, have a look at fx.h with:
#define MIN_SHOW_DELAY 15
Also, in fx_fcn.h, there's:
if (nowUp - _lastShow<MIN_SHOW_DELAY) return;
- Frametime is a constant, which is 1000/42. In order to increase the FPS, instead of return frametime, we can write a smaller value than that and for max speed simply go for return(0), which is the smallest.
Some animations may break when the users start implementing SEGMENTS. Issues encountered were:
- Triggers. A triggered event, which was reset by the animation. This does not work with SEGMENTS. One workaround is knowing that the MIN_SHOW_DELAY is 15, and then determine if now-original_event > MIN_SHOW_DELAY in order to reset the animation.
- To get the current segment being displayed, try Serial.println(_segment_index);
- To use persistent variable across SEGMENTS, don't use 'static', but rather use the existing uint16_t defined SEGENV.aux0 and SEGENV.aux1 variables. Too bad they're not uint32_t.
- For further reading on persistent variables, see this page https://github.com/atuline/WLED/wiki/Persistent-Variables-and-WLED
Here's a replacement for EVERY_N_MILLIS()
uint8_t secondHand = millis()/(256-SEGMENT.speed) % 10;
if (SEGENV.aux0 != secondHand) {
SEGENV.aux0 = secondHand;
<rest of code goes here>
}
In the FastLED world, we could cascade led information with things like:
leds[i+1] = leds[i];
That's not an option with WLED (unless you use the *ledData method described above), because it doesn't have the leds[] array. Instead, you have:
setPixelColor(i+1,getPixelColor(i));
The problem, however is that while the FastLED method preserves the original pixel information, the 'WLED' method is lossy, and eventually the cascaded led's will fade out entirely. The workaround, as previously shown, is to create an array used by the segment that preserves the LED information. AirCoookie's method is to allocate memory on the fly for this. For instance:
if (!SEGENV.allocateData(SEGLEN)) return mode_static(); // allocation failed
SEGENV.data[SEGLEN-1] = 0; // a byte value
If you want, you can give it another name with:
if (!SEGENV.allocateData(SEGLEN)) return mode_static(); // allocation failed
byte* heat = SEGENV.data;
heat[SEGLEN-1] = 0; // a byte value
You now have SEGENV.data[SEGLEN] allocated for your use. Adding a structure for use with your segment is a whole other level of complexity and can be found by examining mode_multi_comet and mode_oscillate among others. Just search for SEGENV.allocateData in FX.cpp.
Introduction
Installing and Compiling
First Time Setup
Running Sound Reactive WLED
2D Support
Sound Settings
2D LED Preferences
ARTI-FX
UDP Sound Sync
Sound Reactive Animations
Non-Reactive Animations
Digital Microphone Hookup
Analog Audio Input Options
Using my PC for the Sound
News
It's Not Working
Noise and Spikes
Connectivity Issues
WLED Programming Notes
Modifying Sound Reactive WLED
Future Directions
Adding a new Settings Page
On Lossy Colours
Sliders in WLED
Testing