Skip to content

hpwit/I2SClocklessVirtualLedDriver

Repository files navigation

build status   Badge Version   Badge Version

I2SClocklessVirtualLedDriver for esp32 and esp32S3

Introduction

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

What kind of led do you want to drive

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.

Why have I rewritten the library ?

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.

Let's start

How can 120 strips can de driven in parallel with an esp32 !!!

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.

Pixel object

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.

MACROS that have nothing to do with led driving

  • 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++;
});

Array of strips

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.

OK, but what is the link between an array of strips and this driver ?

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.

First let's declare a new driver

#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;

How to define my leds array ?

You are using RGB type leds

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

You are using RGBW type leds

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];

Driver functions

void initled(uint8_t *leds, int *Pinsq, int CLOCK_PIN, int LATCH_PIN):

This function initialize the strips.

  • *leds: a pointer to the leds array
  • *Pins: a pointer to the pins array
  • CLOCK_PIN: pin for the clock output needs to be >=16
  • LATCH_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);

Other initialization functions:

  • 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 frameBuffer
  • void 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

For esp32S3 only

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 value
  • clock_1000KHZ : works really fine
  • clock_1111KHZ : depending on your boards
  • clock_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);

setBrightness(int brightness):

This function sets the default brightness for 0->255

setGamma(float gammar, float gammab, float gammag) :

This function sets the gamma of the leds for RGB leds

setPixel(uint32_t pos, uint8_t red, uint8_t green, uint8_t blue):

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

setPixel(uint32_t pos, uint8_t red, uint8_t green, uint8_t blue,uint8_t white):

Set the color of a pixel for RGBW strips

showPixels():

This function displays the pixels.

showPixels(uint8_t *leds):

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);

Let's talk about framerate

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

How do we improve the framerate

On the same core

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.

Enable the use of the second core

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.

▶️ NB: by construct using driver.enableShowPixelsOnCore(1) all the following showPixels() will be like showPixels(NO_WAIT)

Impact of these two previous functions

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

Framebuffer

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();

▶️if you initialized yuour leds with 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

Use of color palettes

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.

moving the palette instead of the pixels

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();

More than 256 colors palette ?

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
}

Leds mappping

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.

How to define the mapping

for this driver you need to map the led number in the X,Y coordinates to the leds number of the strips.

mapping

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;
    }
}

You need to indicate to the driver that you will use mapping

#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

About I2S_MAPPING_MODE_OPTION_MAPPING_IN_MEMORY

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 the setMapLed 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.

▶️ NB: setMapLed(NULL) will cancel the mapping. As you will see in mapping.ino

What if I want to display a iamge larger or smaller than my panel ?

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.

the OffsetDisplay object:

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();.

Defining a panel

#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);
}

What happen if the image is smaller than the panel ?

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

Scrolling, rotaition, scaling

The OffsetDisplay struct has more attributes :

  • offsetx and offsety :
  • scaling : scale the image.try negatiuve number it will revese the image
  • rotation : rotation angle
  • xc and yc : set the rotation center

mapping

About I2S_MAPPING_MODE_OPTION_SCROLL_MAPPING_ALL_IN_MEMORY

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.

Ivideo

Some Optimizations

Artifacts due to interrupts

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)

Reduce the time spent in the buffer creation

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

Increase the buffer length

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.

mapping

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"

mapping

It is corrected

Impact of increasing the DMA buffers

  • 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
  • CON: It decreases the frame rate as each buffer takes longer to be sent

Infinite number of pixels

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

mapping

Two examples pixelpush.ino and pixelpushwithpalette.ino

Why do I say infinte pixels ?

As this method do not require any led memory you can send pixels forever to an infinite set of strips.... 😱 😱

Can I get crazier ? Yes => Interrupt lines

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.

The parameters you can change

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);

While mixing all together you can do something like this: mapping

About

No description, website, or topics provided.

Resources

License

Stars

Watchers

Forks

Sponsor this project

 

Packages

No packages published