Hello led afficionados !! Here is the new version of the Virtual pins library. In reality this version of the library had been sitting more or less finalized on my laptop for a while. I needed to take the time and energy to write the correct examples and of course update the documentation. I have been writing led drivers for the past couple of years now while I was building my 123x48 panel. It inspired me to create the I2S driver implemented in FastLED and then the Virtual pins library. I am also planning to merge all my different led libraries (4 of them in total)
NB : On the ESP32SS3 you can overclock the leds up to 1125KHZ instead of 800KHZ
This library is a new take on driving ws2812 leds with I2S on an esp32. It allows to drive up to 120 strips !!! leds in parallel of
- RGB:
- WS2812,
- WS2813,
- WS2815
- RGBW
- SK6812.
This library is fully compatible with FastLED library objects and its functions.
I have rewritten the library out of the FastLED framework to allow easier testing but also create a pixel pusher independant of the library you want to use. But the main reason is the way I wanted to drive the leds. I have tried to put more functionalities in the driver than the usual 'leds pusher'. Indeed this driver integrates:
- led mapping
- color palette
- ease the use of the second core
- framebuffer
- non blocking capabilities
- 'live led' calculation
- scrolling, rotating, scaling
- duplication
- emulate 'line interrupts' (retro programers will understand)
- Options to avoid artifacts if you have interrupt intensive code
I am trying to be kinda lenghtly on this readme. I hope to explain the why of some functions and for the user to use the one most suitable for its use case. Some of the 'maybe most' of the functionailities will never been used be it was fun and intersting to implement.
The esp32 has 24 pins that can act as outputs pins. This is far away from 120 pins ... The main idea is to drive 8 strips out of one single pin of the ESP32. and you can do this for 15 esp32 pins which gives you 8x15=120 strips !!!
To do this feat you will need to get some ICs
- 74HC245 : this is a bus used as a level shifter (you will need only one of them for LATCH and CLOCK)
- 74HC595 : this is an 8 bit shift register (you will need one 74HC595 for each Virtual pin)
You'll find the schematics within the 'extra' folder of this library.
I personally use it to drive my panels 5904=123x48 (16 strips of 369 leds) and a 12288=128x96 (48 16x16 panels) but still have space for an I2C Nintenda controller and SD card reader.
In FastLED CRGB
and CHSV
objects exist, they represent data in RGB or CHSV color space. These are made of 3 bytes (24 bits)
In this library there is a similar object Pixel
which is equivalent to CRGB
. But it can be changed at compile time.
#define COLOR_RGBW
#include "I2SClocklessVirtualLedDriver.h"
#define NUM_LEDS 200
Pixel leds[NUM_LEDS];
//here the Pixel is not RGB but RGBW hence this time it will be a 36 bits
//it's still ocmpatible with the CRGB object
leds[0]=CRGB(233,23,44);
in that case the following mapping will be done W = min(R, G, B); R = R - W; G = G - W; B = B - W;
In the examples you will find some that integrate the FastLED objects.
HOW_LONG(name, func)
: This macro will the time taken by a part of the code ex:
HOW_LONG("Time taken", {
//insert the code i.e
driver.showPixels();
});
the result in the serial output:
The function *** Time taken *** took 100 ms or 10 fps
RUN_SKETCH_FOR(name, duration, func)
: Execute part of the code for the duration in millisecondes ex:
//it will run for 1000ms or 1s
int offset=0;
RUN_SKETCH_FOR("scroll", 1000 {
leds[offset]=CRGB::Red;
driver.showPixels();
offset++;
});
RUN_SKETCH_N_TIMES(name, ntimes, func)
: Execute part of the code N times ex:
//it will run 100 times
int offset=0;
RUN_SKETCH_FOR("scroll", 100 {
leds[offset]=CRGB::Red;
driver.showPixels();
offset++;
});
In most leds driver librairies you declare each strip attached to one pin, one line at a time.
example 1: For 4 strips in FastLED
CRGB leds1[number_of_leds1];
CRGB leds2[number_of_leds2];
CRGB leds3[number_of_leds3];
CRGB leds4[number_of_leds4];
FasLED.addLeds<PIN1,ORDER>(leds1,number_of_leds1);
FasLED.addLeds<PIN2,ORDER>(leds2,number_of_leds2);
FasLED.addLeds<PIN3,ORDER>(leds3,number_of_leds3);
FasLED.addLeds<PIN4,ORDER>(leds4,number_of_leds4);
example 2: If you are using a large array of same length strips, you would do this:
CRGB leds[4*NUM_LED_PER_STRIPS];
FasLED.addLeds<PIN1,ORDER>(leds,0,NUM_LED_PER_STRIPS);
FasLED.addLeds<PIN2,ORDER>(leds,NUM_LED_PER_STRIPS,NUM_LED_PER_STRIPS);
FasLED.addLeds<PIN3,ORDER>(leds,2*NUM_LED_PER_STRIPS,NUM_LED_PER_STRIPS);
FasLED.addLeds<PIN4,ORDER>(leds,3*NUM_LED_PER_STRIPS,NUM_LED_PER_STRIPS);
example 3: For information if you want to get (for development purpose ease) the leds1,leds2,...
CRGB *leds1=leds;
CRGB *leds2=leds+NUM_LED_PER_STRIPS;
CRGB *leds3=leds+2*NUM_LED_PER_STRIPS;
CRGB *leds4=leds+3*NUM_LED_PER_STRIPS;
If all the strips of the first example are of the same size, then the 2 examples are the doing exactly the same. Hence when using strips of different lengths we cannot put them in a big array ? FALSE. You cant create a large array when using NUM_LED_PER_STRIP
being the largest of number_of_leds
. Of course you array woul be larger than you actual numbre of leds but we can do with the lost of space.
Here is how we would declare the 4 strips in of our example:
CRGB leds[4*NUM_LED_PER_STRIPS];
int pins[4]={PIN1,PIN2,PIN3,PIN4};
driver.initled((uint8_t*)leds,pins,4,NUM_LED_PER_STRIPS,ORDER_GRB);
We are declaring that my leds
array represent 4 strips of NUM_LED_PER_STRIPS
leds ,each strip being linked to the pins defined in the pins array pins
. This is way easier to declare a lot of strips. As discussed before if your strips are not of the same lentgh just define NUM_LED_PER_STRIPS
with the largest number_of_leds
.
#define NBIS2SERIALPINS 2 //the number of virtual pins
//set the strip color order
//by default RGB
/* the other values
#define COLOR_RGB
#define COLOR_RBG
#define COLOR_GBR
#define COLOR_BGR
#define COLOR_BRG
#define COLOR_RGBW
*/
#define NUM_LEDS_PER_STRIP 256
#define NUM_LEDS (NUM_LEDS_PER_STRIP*NBIS2SERIALPINS*8)
#define CLOCK_PIN 16
#define LATCH_PIN 4
#include "I2SClocklessVirtualLedDriver.h"
I2SClocklessVirtualLedDriver driver;
RGB type leds store the information of the led over 3 bytes. Red,Green,Blue
. Hence the size in bytes of a led array of NUM_LEDS
is 3xNUM_LEDS
Pixels leds[NUM_LEDS];
//if you are using FastLED library this definition will be equivalent to the previous as the CRGB object is 3 bytes
CRGB leds[NUM_LEDS];
//you can use either of those
This time to store the information of the led you will need 4 bytes Red,Green,Blue,White
Hence the size in bytes of a led array of NUM_LEDS
is 4xNUM_LEDS
Pixels leds[UM_LEDS];
This function initialize the strips.
*leds
: a pointer to the leds array*Pins
: a pointer to the pins arrayCLOCK_PIN
: pin for the clock output needs to be >=16LATCH_PIN
: pin for the latch ouput
example 4: declaring 16 strips of 256 leds in GRB
#define NBIS2SERIALPINS 2 //the number of virtual pins
#define COLOR_GRB 1 //set the strip color
#define NUM_LEDS_PER_STRIP 256
#define NUM_LEDS (NUM_LEDS_PER_STRIP*NBIS2SERIALPINS*8)
#define CLOCK_PIN 16
#define LATCH_PIN 4
#define NUM_STRIPS (NBIS2SERIALPINS*8)
#include "I2SClocklessVirtualLedDriver.h"
I2SClocklessVirtualLedDriver driver;
Pixels leds[NUM_STRIPS*NUM_LEDS_PER_STRIP]; //equivalent of CRGB leds[NUM_LEDS_PER_STRIPS*NUM_STRIPS]
int pins[NBIS2SERIALPINS] ={0,2};
driver.initled((uint8_t*)leds,pins,CLOCK_PIN,LATCH_PIN);
example 5: declaring 30 strips of 256 leds in RGBW
#define NBIS2SERIALPINS 4 //the number of virtual pins
#define COLOR_GRBW 1 //set the strip color to GRBW
#define NUM_LEDS_PER_STRIP 256
#define NUM_LEDS (NUM_LEDS_PER_STRIP*NBIS2SERIALPINS*8)
#define CLOCK_PIN 16
#define LATCH_PIN 4
#define NUM_STRIPS 30
#include "I2SClocklessVirtualLedDriver.h"
I2SClocklessVirtualLedDriver driver;
Pixel leds[NUM_STRIPS*NUM_LEDS_PER_STRIP];
int pins[NBIS2SERIALPINS] ={0,2,12,13};
driver.initled((uint8_t*)leds,pins,CLOCK_PIN,LATCH_PIN);
void initled(Pixel leds, int *Pinsq, int CLOCK_PIN, int LATCH_PIN)
void initled(CRGB leds, int *Pinsq, int CLOCK_PIN, int LATCH_PIN)
void initled(frameBuffer *leds, int *Pinsq, int CLOCK_PIN, int LATCH_PIN)
this will be explained later on how to use a frameBuffervoid initled(int *Pinsq, int CLOCK_PIN, int LATCH_PIN)
: you can initialized the driver without a led buffer. this is also used in the case of direct pixel calculation
Thanks to the ESP32 LCD driver being faster you can overclock the leds output (which is limited on the esp32 to send 8 pixels per pin at 800KHZ you need a 19.2Mhz clock and the esp32 above 20Mhz is not reliable)
void initled(Pixel leds, int *Pinsq, int CLOCK_PIN, int LATCH_PIN,clock_speed clock)
void initled(CRGB leds, int *Pinsq, int CLOCK_PIN, int LATCH_PIN,clock_speed clock)
void initled((uint8_t*)leds,pins,CLOCK_PIN,LATCH_PIN,clock_speed clock)
Here are the possible values
clock_800KHZ
: default valueclock_1000KHZ
: works really fineclock_1111KHZ
: depending on your boardsclock_1123KHZ
: the max I manage to get
example:
#include "I2SClocklessVirtualLedDriver.h"
I2SClocklessVirtualLedDriver driver;
Pixel leds[NUM_STRIPS*NUM_LEDS_PER_STRIP];
int pins[NBIS2SERIALPINS] ={14, 12, 13, 25};
driver.initled((uint8_t*)leds,pins,CLOCK_PIN,LATCH_PIN,clock_1111KHZ);
Pixel led1[NUM_LEDS];
Pixel leds2[NUM_LEDS];
driver.initled(pins,CLOCK_PIN,LATCH_PIN);
...
driver.showPixels(led1);
...
driver.showPixels(led2);
This function sets the default brightness for 0->255
This function sets the gamma of the leds for RGB leds
Set the color of a pixel NB1: if you are using a RGBW led, this function will do and RGB->RGBW transformation with the following algotithm thanks to @Jonathanese
W = min(R, G, B);
R = R - W;
G = G - W;
B = B - W;
NB: if you are familiar with FastLED it would be leds[pos]=CRGB(red,green,blue)
as you will see in the examples
Set the color of a pixel for RGBW strips
This function displays the pixels.
This function allow set 'on the go' the pointer to the leds. This will help if you are using two buffers for your animation.
It can also be used to ease dithering see example Dithering
(I need to work on a hardware implementation btw)
#define NBIS2SERIALPINS 4 //the number of virtual pins
#define COLOR_GRB 1 //set the strip color
#define NUM_LEDS_PER_STRIP 256
#define NUM_LEDS (NUM_LEDS_PER_STRIP*NBIS2SERIALPINS*8)
#define CLOCK_PIN 16
#define LATCH_PIN 4
#define NUM_STRIPS 30
#include "I2SClocklessVirtualLedDriver.h"
I2SClocklessVirtualLedDriver driver;
Pixel leds[NUM_STRIPS*NUM_LEDS_PER_STRIP];
Pixel leds2[NUM_STRIPS*NUM_LEDS_PER_STRIP];
int pins[NBIS2SERIALPINS] ={0,2,12,13};
driver.initled((uint8_t*)leds,pins,CLOCK_PIN,LATCH_PIN);
//displyaing the leds in leds1
driver.showPixels();
//displaying the leds in leds2
driver.showPixels(leds2);
It's kind of a goal to display as fast as it's possible to have a good framerate. The calculation of the framerate is not only base on the time to display the leds but also the time 'create' the led array. i.e
//display total tiome/fps fpstotal
HOW_LONG("Total framerate", {
//display time/fps to calculate the array fpscalc
HOW_LONG("calcualtion led", {
for(int i=0;i<NUM_LEDS;i++)
{
//do something
}
});
//will display time/fps display the array fpsdisp
HOW_LONG("display", {
driver.showPixels();
});
});
// here is the formula to calculte the total framerate 1/fpstotal=1/fpscalc+1/fpsdisp
Even if the showPixels() takes time, actually the CPU is not busy during the time. Indeed even if to send a pixel (RGB) it takes 30us (microseconds) but the calculation of the buffer takes for 5us to 23us depending on the number of virtual pins to calculate. As a consequence we are loosing CPU time. Hopefully the esp32 rune several tasks at the sametime thanks to RTOS.
the driver allows to take advantage of this quite easily. void showPixels(displayMode dispmode)
......
driver.showPixels(NO_WAIT);
....
In the example waitnowaitmode.ino
we display 256 leds (max framerate 130fps). If you run the code (even without any leds attached) it will display 75fps
for the showPixels() and 112fps
for the showPixels(NO_WAIT)
.
Why is it not 130 fps the max ? because the calculation of the leds array is done wiht the same ressource as the one to calculate the leds buffers to be sent. Both 'compiting' for the same CPU ressource. But we get a 50% speed increase.
The esp32 has two cores and the driver simplify the usage of the second core.
The main core is core 1 on which runs the application bu default. The second core (core 0) is less used. We can tell the driver to run driver.showPixels() on the second core by enableShowPixelsOnCore(int core)
......
//move the display on core 0
driver.enableShowPixelsOnCore(0);
//from now on all the showPixels() will be done on core 0
....
//move the display back on core 1
driver.enableShowPixelsOnCore(1);
In the example secondcore.ino
without the second core we are still at 75fps
and 129fps
(close to the max) when using the two cores.
In the latest case both the calculation and the display occurs at the sametime.
driver.enableShowPixelsOnCore(1)
all the following showPixels()
will be like showPixels(NO_WAIT)
Let's have a look at this example.(here I use NO_WAIT
but I could have used the second core)
for(int i=0;i<NUM_LEDS;i++)
{
leds[i]=CRGB::Red;
}
driver.showPixels(NO_WAIT);
for(int i=0;i<NUM_LEDS;i++)
{
leds[i]=CRGB::Blue;
}
After the showPixel(NO_WAIT)
the code continues its execution and the second loop starts. But we have an issue because the program modifies the leds array that is currently being displayed. In that case some blue leds will appear 😱.
If you're displaying a fast animation this would not be to much of an issue but in any case we need to find a solution if you see any overlap due to this :
- calculate the next frame starting with the pixels that would be displayed last. good luck with your algorithm.
- use two frame buffers
Here it is 😃 As this driver simplify the use of the second core, it simplifies the use of framebuffers.
//use this instead of Pixel leds[NUM_LEDS]
frameBuffer leds = frameBuffer(NUM_LEDS);
...
driver.initled(&leds, Pins, _CLOCK_PIN, _LATCH_PIN);
//NB: it's initled(&leds,... this time and not initled(led,...
The rest of your code will not change at all
for (int i = 0; i < NBIS2SERIALPINS * 16; i++)
{
for (int j = 0; j < 8 * 16; j++)
{
//here we write on buffer x
leds[i * (8 * 16) + (j) % (8 * 16)] = colors[offset % 3];
}
}
//we display buffer x, we change the writing buffer to x+1
driver.showPixels();
//now we write in buffer x+1
leds[54]=CRGB(25,48,79);
//we display buffer x+1 , we switch back the writing buffer to x
driver.showPixels();
framebuffer
Automatically upon calling showPixels
:
- The call will be switch to
NO_WAIT
- the driver will change the buffer on which you be writing
hence in the following example at each iteration of the loop we write in another buffer that the one we are currently displaying.
void loop()
{
for(int i=0;i<NUM_LEDS;i++)
{
//do something with the leds
}
driver.showPixels();
}
In the example framebuffer.ino
I advice you to change #define TEST_USE_FRAMEBUFFER 1
to #define TEST_USE_FRAMEBUFFER 0
to see the effect of the overlap
Even if the RBG leds allows you to display more than 16 millions colors, it can happen that you do not need that many colors. you would refer to a palette then.
for instance if you have less than 256 colors, instead of having Pixel leds[NUM_LED]
which take 3xNUM_LED
bytes you can define uint8_t leds[NUM_LED]
whihc will take 3 times less memory.
#define _USE_PALETTE //necessary to tell the driver to use palette
#include "I2SClocklessVirtualLedDriver.h"
...
Pixel palette[256]; //or less if you do not use all 256 colors.
uint8_t leds[NUM_LEDS];
...
void functionToFillThePalette(param ..)
{
//do so
}
void setup()
{
driver.initled(leds, Pins, CLOCK_PIN, LATCH_PIN);
driver.setPalette((uint8_t *)palette);
}
...
driver.showPixels();
NB: the mapping between the leds value and the palette color is done at runtime => no need of extra memory. There is no creation od a temporary led array.
Like in our old game console, soem animation were made by switching palette. This is waht will happen if you change the palette. like in the example colorpalete.ino
.
In this case the animation is occuring because of the swift of the palette colors. I need to change 256 color valuies instead of recalculating 12000 leds. In this exmaple I am modifying the palette bu of course you could do this:
Pixel palette1[256];
Pixel palette2[256];
...
driver.setPalette((uint8_t *)palette1);
driver.showPixels();
driver.setPalette((uint8_t *)palette2);
driver.showPixels();
You can create a 2bytes palette for a max of 65536 colors. in that case your leds array would need to be 2 bytes per leds
#define _USE_PALETTE //necessary to tell the driver to use palette
#define PALETTE_SIZE 2 //for a 2 bytes palette
#include "I2SClocklessVirtualLedDriver.h"
Pixel palette[11123];
uint16_t leds[NUM_LEDS]; //still 1 third less memory than RGB led array
void setup()
{
driver.initled(leds, Pins, CLOCK_PIN, LATCH_PIN);
driver.setPalette((uint8_t *)palette); //still the same call to match the palette
}
When creating leds structures sometimes it can be hard to match how the leds informaiton is stored and what it physically represent. For instance you have a panel with a snake pattern or a set of 16x16 panels or something else. You can always write a map(x,y) function but the default is that you cannot manipulate the memory as you wish. THe mapping functions do not work for pictures or for artnet you need to reproduce the physical representation of the strip in your artnet software. The idea of the driver is to provide a way to define the mapping function so that in memory the leds are laid like your expect in (X,Y) cooridinate. the acutal mapping is done using calculation withoout touching the led array. Hence for instance you can define your artnet universes as a simple rectangle and it will display correctly.
for this driver you need to map the led number in the X,Y coordinates to the leds number of the strips.
Here is the correspondant mapping function.
uint16_t mapfunction(int pos)
{
int x=pos%8;
int y=pos/8;
if(y%2 == 0)
{
return y*8+x;
}
else
{
return (y+1)*8-x-1;
}
}
#define I2S_MAPPING_MODE I2S_MAPPING_MODE_OPTION_MAPPING_IN_MEMORY
#include "I2SClocklessVirtualLedDriver.h"
#define LED_HEIGHT 45
#define LED_WIDTH 14
...
uint16_t mapfunction(int pos)
{
//your maapping function
}
....
driver.initled(leds, Pins, CLOCK_PIN, LATCH_PIN);
driver.setMapLed(&mapfunction);
...
for (int i=0;i<LED_WIDTH;i++)
{
for(int j=0;j<LED_HEIGHT;j++)
{
leds[j*LED_WIDTH+i]=.... //here you're putting your leds in the usual X,Y coordinates
}
}
driver.showPixels();
Because the memory layout is exactly the one of the X,Y coordinate it makes manipulation easier and faster than doing leds[map(i,j)]=...
please find a full code example in mapping.ino
There are two 'technical' options for the driver to deal with the 'hardware' mapping.
- In memory
#define I2S_MAPPING_MODE I2S_MAPPING_MODE_OPTION_MAPPING_IN_MEMORY
: once thesetMapLed
is executed, an array for the correspondance is created- PRO: less CPU intensive
- CON: you need more memory
- In software
#define I2S_MAPPING_MODE I2S_MAPPING_MODE_OPTION_MAPPING_SOFTWARE
: the mapping function is call during the interrupt function that creates the memory buffer.- PRO: no need for more memory
- CON: more CPU intensive
Depending on your specific need you can choose the option that is more suitable.
setMapLed(NULL)
will cancel the mapping. As you will see in mapping.ino
Let's imagine that you want to display a 200x200 picture on a 32x32 led panel. You need to create an algortihm and then if you want to make thispicture scroll it's even more complicated. This driver allows you to do this with simple commands and even more.
struct OffsetDisplay
{
int offsetx;
int offsety;
int panel_height;
int panel_width;
int image_height;
int image_width;
float scalling;
int xc;
int yc;
float rotation;
bool enableLoopx;
bool enableLoopy;
....
};
At the initiation of the leds a default Offdisplay is created with certain values. You can get this default object with getDefaultOffset();
.
#define NBIS2SERIALPINS 6
#define NUM_LEDS_PER_STRIPS 256
#define PICTURE_WIDTH 256
#define PICTURE_HEIGHT 256
#define I2S_MAPPING_MODE I2S_MAPPING_MODE_OPTION_SCROLL_MAPPING_ALL_IN_MEMORY //to activate the option needed
#include "I2SClocklessVirtualLedDriver.h"
#include "pics.h" //which contains the image
I2SClocklessVirtualLedDriver driver;
OffsetDisplay offd;
uint16_t mapfunction(uint16_t pos)
{
int panelnumber = pos / 256;
int datainpanel = pos % 256;
int yp = panelnumber / 8;
int Xp = panelnumber % 8;
int Y = yp;
int X = Xp;
int x = datainpanel % 16;
int y = datainpanel / 16;
if (y % 2 == 0)
{
Y = Y * 16 + y;
X = X * 16 + x;
}
else
{
Y = Y * 16 + y;
X = X * 16 + 16 - x - 1;
}
return Y * 16 * 8 + X;
}
void setup()
{
// put your setup code here, to run once:
Serial.begin(115200);
driver.initled(picture, Pins, CLOCK_PIN, LATCH_PIN); //picture is the array of data in the pics.h
driver.setMapLed(&mapfunction);
driver.setBrightness(30);
offd = driver.getDefaultOffset(); //do not forget this
offd.panel_width = 128; //the physical width of the panel
offd.panel_height = 96; //the physical hieght of the panel
offd.image_height = PICTURE_HEIGHT;
offd.image_width = PICTURE_WIDTH;
driver.showPixels(offd);
}
In the case of image smaller image than the panel the none existing pixels wxill be replaced by the data just after the image
Pixel image[PICTURE_HEIGHT*PICTURE_WIDTH+1];
image[PICTURE_HEIGHT*PICTURE_WIDTH]=CRGB::Red;
offd = driver.getDefaultOffset(); //do not forget this
offd.panel_width = 128; //the physical width of the panel
offd.panel_height = 96; //the physical hieght of the panel
offd.image_height = PICTURE_HEIGHT;
offd.image_width = PICTURE_WIDTH;
driver.showPixels(offd);
The image will display by the 'background' will be red
You can duplicate the image in X and Y direction
offd.enableLoopx = true; //the image will duplicate in X direction
offd.enableLoopy = true; //the imade will duplication in Y direction
The OffsetDisplay
struct has more attributes :
offsetx
andoffsety
:scaling
: scale the image.try negatiuve number it will revese the imagerotation
: rotation anglexc
andyc
: set the rotation center
To do the scrolling it's more or less the same as the mapping you have several options
I2S_MAPPING_MODE_OPTION_SCROLL_MAPPING_ALL_IN_MEMORY
: you need two array one for the mapping the second for the scroll- PRO : really fast during the led buffers creation (good if you have a lot of strips > 50)
- CONS : uses a lot of memory; and the framerate is a bit reduced to calculate the SCROLL array
I2S_MAPPING_MODE_OPTION_SCROLL_MAPPING_IN_MEMORY_SOFTWARE
: only one big array for the mapping- PRO : uses less memory; still efficient
- CONS : more CPU intensive than the previous
I2S_MAPPING_MODE_OPTION_SCROLL_MAPPING_SOFTWARE_SOFTWARE
: everyting is done at runtime- PRO : no memory overhead
- CONS : really CPU intensive (see chapter on optimizations)
I2S_MAPPING_MODE_OPTION_SCROLL_MAPPING_SOFTWARE_IN_MEMORY
: the mapping is in software pour the scroll arry is precalculated- PRO : uses less memory; really fast
- CONS : still uses memeory ,overhead during the calculation.
The right option will depend on what the rest of your appplication is doing. for exmaple look at this video while the esp32 is receiving up to 73 universes and doign srcolling and rotations.
Sometimes interrupts can distrub the pixel buffer calculations hence making some artifacts. A solution against that is to caculate several buffers in advance. BY defualt we have 2 dma buffers. this can be increase to cope with unwanted interupts.
#define __NB_DMA_BUFFER 4 //here we increase the number of buffers from 2 to 4 increase this number as much as you need
....
#include "I2SClocklessVirtualLedDriver.h"
NB:If you define __NB_DMA_BUFFER
to be equal to the number of led per strip you will calculate all the leds in a buffer (but it requires to much memory)
If you remember when I have discussed about the fact that the showPixels is not always occupied with gives time for other processes to run. Well the less time we 'spent' in buffer calcualtion the better.for instance if you do not use gamma calculation and you can cope with a brightness that is a power of 2:
#define __BRIGHTNESS_BIT 5 //the max brightness will be 2^5=32
....
#include "I2SClocklessVirtualLedDriver.h"
it will drastically decrease the time of the buffer calculation
Why on earth to that ??????!!!! Sometimes despite the above improvement the time to calculate the buffer is longer the time needed to send the pixel hence the leds are not corrected and look duplicated. The ws281X leds do not need to receive all the pixel one after the other without pause. If you look at the data sheet you will see that if you wait less than 150us than the led will pass the new data like if it was sent just after.
If you activate the Verbose mode while uplaoding your sktech you will have this in the serial output:
7:46:55.946 -> Frame data:
17:46:55.946 -> - frame number:87
17:46:55.946 -> - interupt time min:32.39us
17:46:55.946 -> - interupt time max:33.38us
17:46:55.946 -> - interupt time average:32.71us
17:46:55.946 -> - nb of pixel with interuptime > 26.00us: 255
17:46:55.946 -> Driver data (overall frames):
17:46:55.946 -> - nb of frames displayed:87
17:46:55.946 -> - nb of frames with pixels 'out of time':87
17:46:55.946 -> - max interuptime 34.17us
17:46:55.946 -> - max number of pixels out of interuptime in a frame:255
17:46:55.978 -> - proposed DMA extension:78
by adding this
#define _DMA_EXTENSTION 78
...
#include "I2SClocklessVirtualLedDriver.h"
It is corrected
- PRO:
- If you're using
I2S_MAPPING_MODE_OPTION_SCROLL_MAPPING_SOFTWARE_SOFTWARE
than you can manage to avoid artifacts - it cvan be useful if you have several tasks in parallel on the same core to give other takss more time to execute as extending the DMA buffer has no impact on CPU usage
- If you're using
- CON: It decreases the frame rate as each buffer takes longer to be sent
The driver can act as a pixels pusher. In that case the value of the leds are calculated at runtime without having to store any led array.
#define I2S_MAPPING_MODE I2S_MAPPING_MODE_OPTION_DIRECT_CALCULATION //to activate the pixelpousher mode
#include "I2SClocklessVirtualLedDriver.h"
Pixel functionCalc(uint16_t ledtodisp, int pin, int virtualpin)
{
//calculate the pixels depending on it's position
}
void setup()
{
Serial.begin(115200);
driver.setPixelCalc(&functionCalc);
driver.initled(Pins, CLOCK_PIN, LATCH_PIN); //nb: no need for a led pointer in the initled
driver.showPixels();
}
What can be done can be quite complex
Two examples pixelpush.ino
and pixelpushwithpalette.ino
As this method do not require any led memory you can send pixels forever to an infinite set of strips.... 😱 😱
Yes if I can 😊. I just got a GameBoy couple of weeks ago (yes at my advance age). And I am a fan on demo programming on retro platforms. One of the common trick is to change the displayed picture based on line scan interrupts. I thought it coudl be fun to have this also on the driver.
At each line you can change:
- the offsetx
- the scalingx
- the scalingy
See example interruptlines.ino
#define I2S_MAPPING_MODE (I2S_MAPPING_MODE_OPTION_SCROLL_MAPPING_ALL_IN_MEMORY | I2S_MAPPING_MODE_OPTION_INTERRUPT_LINE) // to activate the interrupts
#include "I2SClocklessVirtualLedDriver.h"
...
driver.offsetsx[i]=...;
driver.scalingx[i]=...;
driver.scalingy[i]=....;
driver.showPixels(offd);