'Final' changes.
Allocate memory at the row level.
This commit is contained in:
parent
33feae2b04
commit
7dd49d8b3c
3 changed files with 101 additions and 194 deletions
|
@ -77,7 +77,7 @@ bool RGB64x32MatrixPanel_I2S_DMA::allocateDMAmemory()
|
||||||
size_t _dma_linked_list_memory_required = 0;
|
size_t _dma_linked_list_memory_required = 0;
|
||||||
size_t _total_dma_capable_memory_reserved = 0;
|
size_t _total_dma_capable_memory_reserved = 0;
|
||||||
|
|
||||||
// 1. Calculate and malloc the LARGEST available DMA memory block to matrix_framebuffer_malloc_1
|
// 1. Calculate the amount of DMA capable memory that's actually available
|
||||||
#if SERIAL_DEBUG
|
#if SERIAL_DEBUG
|
||||||
Serial.printf("Panel Height: %d pixels.\r\n", MATRIX_HEIGHT);
|
Serial.printf("Panel Height: %d pixels.\r\n", MATRIX_HEIGHT);
|
||||||
Serial.printf("Panel Width: %d pixels.\r\n", MATRIX_WIDTH);
|
Serial.printf("Panel Width: %d pixels.\r\n", MATRIX_WIDTH);
|
||||||
|
@ -90,65 +90,38 @@ bool RGB64x32MatrixPanel_I2S_DMA::allocateDMAmemory()
|
||||||
heap_caps_print_heap_info(MALLOC_CAP_DMA);
|
heap_caps_print_heap_info(MALLOC_CAP_DMA);
|
||||||
|
|
||||||
Serial.printf("We're going to need %d bytes of SRAM just for the frame buffer(s).\r\n", _frame_buffer_memory_required);
|
Serial.printf("We're going to need %d bytes of SRAM just for the frame buffer(s).\r\n", _frame_buffer_memory_required);
|
||||||
|
Serial.printf("The total amount of DMA capable SRAM memory is %d bytes.\r\n", heap_caps_get_free_size(MALLOC_CAP_DMA));
|
||||||
Serial.printf("Largest DMA capable SRAM memory block is %d bytes.\r\n", heap_caps_get_largest_free_block(MALLOC_CAP_DMA));
|
Serial.printf("Largest DMA capable SRAM memory block is %d bytes.\r\n", heap_caps_get_largest_free_block(MALLOC_CAP_DMA));
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
// Can we fit the framebuffer into the single DMA capable memory block available?
|
// Can we potentially fit the framebuffer into the DMA capable memory that's available?
|
||||||
if ( heap_caps_get_largest_free_block(MALLOC_CAP_DMA) < _frame_buffer_memory_required ) {
|
if ( heap_caps_get_free_size(MALLOC_CAP_DMA) < _frame_buffer_memory_required ) {
|
||||||
|
|
||||||
#if SERIAL_DEBUG
|
#if SERIAL_DEBUG
|
||||||
Serial.printf("######### Insufficient memory for requested resolution. Reduce MATRIX_COLOR_DEPTH and try again.\r\n\tAdditional %d bytes of memory required.\r\n\r\n", (_frame_buffer_memory_required-heap_caps_get_largest_free_block(MALLOC_CAP_DMA)) );
|
Serial.printf("######### Insufficient memory for requested resolution. Reduce MATRIX_COLOR_DEPTH and try again.\r\n\tAdditional %d bytes of memory required.\r\n\r\n", (_frame_buffer_memory_required-heap_caps_get_free_size(MALLOC_CAP_DMA)) );
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Alright, theoretically we should be OK, so let us do this, so
|
||||||
|
// lets allocate a chunk of memory for each row (a row could span multiple panels if chaining is in place)
|
||||||
|
for (int malloc_num =0; malloc_num < ROWS_PER_FRAME; malloc_num++)
|
||||||
|
{
|
||||||
|
matrix_row_framebuffer_malloc[malloc_num] = (rowColorDepthStruct *)heap_caps_malloc( (sizeof(rowColorDepthStruct) * _num_frame_buffers) , MALLOC_CAP_DMA);
|
||||||
|
// If the ESP crashes here, then we must have a horribly fragmented memory space, or trying to allocate a ludicrous resolution.
|
||||||
|
#if SERIAL_DEBUG
|
||||||
|
Serial.printf("Malloc'ing %d bytes of memory @ address %d for frame row %d.\r\n", (sizeof(rowColorDepthStruct) * _num_frame_buffers), matrix_row_framebuffer_malloc[malloc_num], malloc_num);
|
||||||
|
#endif
|
||||||
|
if ( matrix_row_framebuffer_malloc[malloc_num] == NULL ) {
|
||||||
|
Serial.printf("ERROR: Couldn't malloc matrix_row_framebuffer %d! Critical fail.\r\n", malloc_num);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Allocate the framebuffer 1 memory, fail if we can even do this
|
|
||||||
matrix_framebuffer_malloc_1 = (frameStruct *)heap_caps_malloc(_frame_buffer_memory_required, MALLOC_CAP_DMA);
|
|
||||||
if ( matrix_framebuffer_malloc_1 == NULL ) {
|
|
||||||
Serial.println("ERROR: Couldn't malloc matrix_framebuffer_malloc_1! Critical fail.\r\n");
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
_total_dma_capable_memory_reserved += _frame_buffer_memory_required;
|
_total_dma_capable_memory_reserved += _frame_buffer_memory_required;
|
||||||
|
|
||||||
// SPLIT MEMORY MODE
|
|
||||||
#ifdef SPLIT_MEMORY_MODE
|
|
||||||
|
|
||||||
#if SERIAL_DEBUG
|
|
||||||
Serial.println("SPLIT MEMORY MODE ENABLED!");
|
|
||||||
Serial.print("Rows per frame (overall): "); Serial.println(ROWS_PER_FRAME, DEC);
|
|
||||||
Serial.print("Rows per split framebuffer malloc: "); Serial.println(SPLIT_MEMORY_ROWS_PER_FRAME, DEC);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
|
|
||||||
// Can we fit the framebuffer into the single DMA capable memory block available?
|
|
||||||
// Can we fit the framebuffer into the single DMA capable memory block available?
|
|
||||||
if ( heap_caps_get_largest_free_block(MALLOC_CAP_DMA) < _frame_buffer_memory_required ) {
|
|
||||||
|
|
||||||
#if SERIAL_DEBUG
|
|
||||||
Serial.printf("######### Insufficient memory for second framebuffer for requested resolution. Reduce MATRIX_COLOR_DEPTH and try again.\r\n\tAdditional %d bytes of memory required.\r\n\r\n", (_frame_buffer_memory_required-heap_caps_get_largest_free_block(MALLOC_CAP_DMA)) );
|
|
||||||
#endif
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Allocate the framebuffer 2 memory, fail if we can even do this
|
|
||||||
matrix_framebuffer_malloc_2 = (frameStruct *)heap_caps_malloc(_frame_buffer_memory_required, MALLOC_CAP_DMA);
|
|
||||||
if ( matrix_framebuffer_malloc_2 == NULL ) {
|
|
||||||
Serial.println("ERROR: Couldn't malloc matrix_framebuffer_malloc_2! Critical fail.\r\n");
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
_total_dma_capable_memory_reserved += _frame_buffer_memory_required;
|
|
||||||
|
|
||||||
|
|
||||||
#endif
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
/***
|
/***
|
||||||
* Step 2: Calculate the amount of memory required for the DMA engine's linked list descriptors.
|
* Step 2: Calculate the amount of memory required for the DMA engine's linked list descriptors.
|
||||||
|
@ -156,7 +129,7 @@ bool RGB64x32MatrixPanel_I2S_DMA::allocateDMAmemory()
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|
||||||
// Calculate what color depth is actually possible based on memory avaialble vs. required dma linked-list descriptors.
|
// Calculate what colour depth is actually possible based on memory available vs. required dma linked-list descriptors.
|
||||||
// aka. Calculate the lowest LSBMSB_TRANSITION_BIT value that will fit in memory
|
// aka. Calculate the lowest LSBMSB_TRANSITION_BIT value that will fit in memory
|
||||||
int numDMAdescriptorsPerRow = 0;
|
int numDMAdescriptorsPerRow = 0;
|
||||||
lsbMsbTransitionBit = 0;
|
lsbMsbTransitionBit = 0;
|
||||||
|
@ -282,12 +255,14 @@ bool RGB64x32MatrixPanel_I2S_DMA::allocateDMAmemory()
|
||||||
|
|
||||||
Serial.printf("*** ESP32-RGB64x32MatrixPanel-I2S-DMA: Memory Allocations Complete *** \r\n");
|
Serial.printf("*** ESP32-RGB64x32MatrixPanel-I2S-DMA: Memory Allocations Complete *** \r\n");
|
||||||
Serial.printf("Total memory that was reserved: %d kB.\r\n", _total_dma_capable_memory_reserved/1024);
|
Serial.printf("Total memory that was reserved: %d kB.\r\n", _total_dma_capable_memory_reserved/1024);
|
||||||
|
Serial.printf("... of which was used for the DMA Linked List(s): %d kB.\r\n", _dma_linked_list_memory_required/1024);
|
||||||
|
|
||||||
Serial.printf("Heap Memory Available: %d bytes total. Largest free block: %d bytes.\r\n", heap_caps_get_free_size(0), heap_caps_get_largest_free_block(0));
|
Serial.printf("Heap Memory Available: %d bytes total. Largest free block: %d bytes.\r\n", heap_caps_get_free_size(0), heap_caps_get_largest_free_block(0));
|
||||||
Serial.printf("General RAM Available: %d bytes total. Largest free block: %d bytes.\r\n", heap_caps_get_free_size(MALLOC_CAP_DEFAULT), heap_caps_get_largest_free_block(MALLOC_CAP_DEFAULT));
|
Serial.printf("General RAM Available: %d bytes total. Largest free block: %d bytes.\r\n", heap_caps_get_free_size(MALLOC_CAP_DEFAULT), heap_caps_get_largest_free_block(MALLOC_CAP_DEFAULT));
|
||||||
|
|
||||||
|
|
||||||
#if SERIAL_DEBUG
|
#if SERIAL_DEBUG
|
||||||
Serial.println("DMA memory blocks available after malloc's: ");
|
Serial.println("DMA capable memory map available after malloc's: ");
|
||||||
heap_caps_print_heap_info(MALLOC_CAP_DMA);
|
heap_caps_print_heap_info(MALLOC_CAP_DMA);
|
||||||
delay(1000);
|
delay(1000);
|
||||||
#endif
|
#endif
|
||||||
|
@ -319,21 +294,10 @@ void RGB64x32MatrixPanel_I2S_DMA::configureDMA(int r1_pin, int g1_pin, int b1_
|
||||||
}
|
}
|
||||||
|
|
||||||
// Fill DMA linked lists for both frames (as in, halves of the HUB75 panel) and if double buffering is enabled, link it up for both buffers.
|
// Fill DMA linked lists for both frames (as in, halves of the HUB75 panel) and if double buffering is enabled, link it up for both buffers.
|
||||||
for(int j = 0; j < ROWS_PER_FRAME; j++) {
|
for(int row = 0; row < ROWS_PER_FRAME; row++) {
|
||||||
|
|
||||||
// Split framebuffer malloc hack 'improvement'
|
// Split framebuffer malloc hack 'improvement'
|
||||||
frameStruct *fb_malloc_ptr = matrix_framebuffer_malloc_1;
|
rowColorDepthStruct *fb_malloc_ptr = matrix_row_framebuffer_malloc[row];
|
||||||
int fb_malloc_j = j;
|
|
||||||
|
|
||||||
#ifdef SPLIT_MEMORY_MODE
|
|
||||||
if ( j >= SPLIT_MEMORY_ROWS_PER_FRAME ) {
|
|
||||||
fb_malloc_ptr = matrix_framebuffer_malloc_2;
|
|
||||||
fb_malloc_j -= SPLIT_MEMORY_ROWS_PER_FRAME; // IMPORTANT
|
|
||||||
#if SERIAL_DEBUG
|
|
||||||
Serial.printf("Split memory mode. Panel row: %d, mapped to framebuffer malloc 2 offset: %d.\r\n", j, fb_malloc_j);
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#if SERIAL_DEBUG
|
#if SERIAL_DEBUG
|
||||||
Serial.printf("DMA payload of %d bytes. DMA_MAX is %d.\r\n", sizeof(rowBitStruct) * PIXEL_COLOR_DEPTH_BITS, DMA_MAX);
|
Serial.printf("DMA payload of %d bytes. DMA_MAX is %d.\r\n", sizeof(rowBitStruct) * PIXEL_COLOR_DEPTH_BITS, DMA_MAX);
|
||||||
|
@ -342,11 +306,11 @@ void RGB64x32MatrixPanel_I2S_DMA::configureDMA(int r1_pin, int g1_pin, int b1_
|
||||||
|
|
||||||
// first set of data is LSB through MSB, single pass (IF TOTAL SIZE < DMA_MAX) - all color bits are displayed once, which takes care of everything below and inlcluding LSBMSB_TRANSITION_BIT
|
// first set of data is LSB through MSB, single pass (IF TOTAL SIZE < DMA_MAX) - all color bits are displayed once, which takes care of everything below and inlcluding LSBMSB_TRANSITION_BIT
|
||||||
// NOTE: size must be less than DMA_MAX - worst case for library: 16-bpp with 256 pixels per row would exceed this, need to break into two
|
// NOTE: size must be less than DMA_MAX - worst case for library: 16-bpp with 256 pixels per row would exceed this, need to break into two
|
||||||
link_dma_desc(&dmadesc_a[current_dmadescriptor_offset], previous_dmadesc_a, &(fb_malloc_ptr[0].rowdata[fb_malloc_j].rowbits[0].data), sizeof(rowBitStruct) * num_dma_payload_color_depths);
|
link_dma_desc(&dmadesc_a[current_dmadescriptor_offset], previous_dmadesc_a, &(fb_malloc_ptr[0].rowbits[0].data), sizeof(rowBitStruct) * num_dma_payload_color_depths);
|
||||||
previous_dmadesc_a = &dmadesc_a[current_dmadescriptor_offset];
|
previous_dmadesc_a = &dmadesc_a[current_dmadescriptor_offset];
|
||||||
|
|
||||||
if (double_buffering_enabled) {
|
if (double_buffering_enabled) {
|
||||||
link_dma_desc(&dmadesc_b[current_dmadescriptor_offset], previous_dmadesc_b, &(fb_malloc_ptr[1].rowdata[fb_malloc_j].rowbits[0].data), sizeof(rowBitStruct) * num_dma_payload_color_depths);
|
link_dma_desc(&dmadesc_b[current_dmadescriptor_offset], previous_dmadesc_b, &(fb_malloc_ptr[1].rowbits[0].data), sizeof(rowBitStruct) * num_dma_payload_color_depths);
|
||||||
previous_dmadesc_b = &dmadesc_b[current_dmadescriptor_offset]; }
|
previous_dmadesc_b = &dmadesc_b[current_dmadescriptor_offset]; }
|
||||||
|
|
||||||
current_dmadescriptor_offset++;
|
current_dmadescriptor_offset++;
|
||||||
|
@ -363,11 +327,11 @@ void RGB64x32MatrixPanel_I2S_DMA::configureDMA(int r1_pin, int g1_pin, int b1_
|
||||||
{
|
{
|
||||||
// first set of data is LSB through MSB, single pass - all color bits are displayed once, which takes care of everything below and inlcluding LSBMSB_TRANSITION_BIT
|
// first set of data is LSB through MSB, single pass - all color bits are displayed once, which takes care of everything below and inlcluding LSBMSB_TRANSITION_BIT
|
||||||
// TODO: size must be less than DMA_MAX - worst case for library: 16-bpp with 256 pixels per row would exceed this, need to break into two
|
// TODO: size must be less than DMA_MAX - worst case for library: 16-bpp with 256 pixels per row would exceed this, need to break into two
|
||||||
link_dma_desc(&dmadesc_a[current_dmadescriptor_offset], previous_dmadesc_a, &(fb_malloc_ptr[0].rowdata[fb_malloc_j].rowbits[cd].data), sizeof(rowBitStruct) );
|
link_dma_desc(&dmadesc_a[current_dmadescriptor_offset], previous_dmadesc_a, &(fb_malloc_ptr[0].rowbits[cd].data), sizeof(rowBitStruct) );
|
||||||
previous_dmadesc_a = &dmadesc_a[current_dmadescriptor_offset];
|
previous_dmadesc_a = &dmadesc_a[current_dmadescriptor_offset];
|
||||||
|
|
||||||
if (double_buffering_enabled) {
|
if (double_buffering_enabled) {
|
||||||
link_dma_desc(&dmadesc_b[current_dmadescriptor_offset], previous_dmadesc_b, &(fb_malloc_ptr[1].rowdata[fb_malloc_j].rowbits[cd].data), sizeof(rowBitStruct) );
|
link_dma_desc(&dmadesc_b[current_dmadescriptor_offset], previous_dmadesc_b, &(fb_malloc_ptr[1].rowbits[cd].data), sizeof(rowBitStruct) );
|
||||||
previous_dmadesc_b = &dmadesc_b[current_dmadescriptor_offset]; }
|
previous_dmadesc_b = &dmadesc_b[current_dmadescriptor_offset]; }
|
||||||
|
|
||||||
current_dmadescriptor_offset++;
|
current_dmadescriptor_offset++;
|
||||||
|
@ -389,11 +353,11 @@ void RGB64x32MatrixPanel_I2S_DMA::configureDMA(int r1_pin, int g1_pin, int b1_
|
||||||
|
|
||||||
for(int k=0; k < (1<<(i - lsbMsbTransitionBit - 1)); k++)
|
for(int k=0; k < (1<<(i - lsbMsbTransitionBit - 1)); k++)
|
||||||
{
|
{
|
||||||
link_dma_desc(&dmadesc_a[current_dmadescriptor_offset], previous_dmadesc_a, &(fb_malloc_ptr[0].rowdata[fb_malloc_j].rowbits[i].data), sizeof(rowBitStruct) * (PIXEL_COLOR_DEPTH_BITS - i));
|
link_dma_desc(&dmadesc_a[current_dmadescriptor_offset], previous_dmadesc_a, &(fb_malloc_ptr[0].rowbits[i].data), sizeof(rowBitStruct) * (PIXEL_COLOR_DEPTH_BITS - i));
|
||||||
previous_dmadesc_a = &dmadesc_a[current_dmadescriptor_offset];
|
previous_dmadesc_a = &dmadesc_a[current_dmadescriptor_offset];
|
||||||
|
|
||||||
if (double_buffering_enabled) {
|
if (double_buffering_enabled) {
|
||||||
link_dma_desc(&dmadesc_b[current_dmadescriptor_offset], previous_dmadesc_b, &(fb_malloc_ptr[1].rowdata[fb_malloc_j].rowbits[i].data), sizeof(rowBitStruct) * (PIXEL_COLOR_DEPTH_BITS - i));
|
link_dma_desc(&dmadesc_b[current_dmadescriptor_offset], previous_dmadesc_b, &(fb_malloc_ptr[1].rowbits[i].data), sizeof(rowBitStruct) * (PIXEL_COLOR_DEPTH_BITS - i));
|
||||||
previous_dmadesc_b = &dmadesc_b[current_dmadescriptor_offset]; }
|
previous_dmadesc_b = &dmadesc_b[current_dmadescriptor_offset]; }
|
||||||
|
|
||||||
current_dmadescriptor_offset++;
|
current_dmadescriptor_offset++;
|
||||||
|
@ -481,16 +445,7 @@ void RGB64x32MatrixPanel_I2S_DMA::updateMatrixDMABuffer(int16_t x_coord, int16_t
|
||||||
red = lumConvTab[red];
|
red = lumConvTab[red];
|
||||||
green = lumConvTab[green];
|
green = lumConvTab[green];
|
||||||
blue = lumConvTab[blue];
|
blue = lumConvTab[blue];
|
||||||
/*
|
|
||||||
red = val2PWM(red);
|
|
||||||
green = val2PWM(green);
|
|
||||||
blue = val2PWM(blue);
|
|
||||||
*/
|
|
||||||
#ifdef SPLIT_MEMORY_MODE
|
|
||||||
#ifdef SERIAL_DEBUG
|
|
||||||
int tmp_y_coord = y_coord;
|
|
||||||
#endif
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/* 1) Check that the co-ordinates are within range, or it'll break everything big time.
|
/* 1) Check that the co-ordinates are within range, or it'll break everything big time.
|
||||||
* Valid co-ordinates are from 0 to (MATRIX_XXXX-1)
|
* Valid co-ordinates are from 0 to (MATRIX_XXXX-1)
|
||||||
|
@ -499,37 +454,42 @@ void RGB64x32MatrixPanel_I2S_DMA::updateMatrixDMABuffer(int16_t x_coord, int16_t
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* 2) Convert the vertical axis / y-axis pixel co-ordinate to a matrix panel parallel co-ordinate..
|
/* When using the drawPixel, we are obviously only changing the value of one x,y position,
|
||||||
* eg. If the y co-ordinate is 23, that's actually in the second half of the panel, row 7.
|
* however, the two-scan panels paint TWO lines at the same time
|
||||||
* 23 (y coord) - 16 (for 32px high panel) = 7
|
* and this reflects the parallel in-DMA-memory data structure of uint16_t's that are getting
|
||||||
*/
|
* pumped out at high speed.
|
||||||
bool paint_top_half = true;
|
*
|
||||||
|
* So we need to ensure we persist the bits (8 of them) of the uint16_t for the row we aren't changing.
|
||||||
|
*
|
||||||
|
* The DMA buffer order has also been reversed (refer to the last code in this function)
|
||||||
|
* so we have to check for this and check the correct position of the MATRIX_DATA_STORAGE_TYPE
|
||||||
|
* data.
|
||||||
|
*/
|
||||||
|
bool painting_top_frame = true;
|
||||||
if ( y_coord >= ROWS_PER_FRAME) // co-ords start at zero, y_coord = 15 = 16 (rows per frame)
|
if ( y_coord >= ROWS_PER_FRAME) // co-ords start at zero, y_coord = 15 = 16 (rows per frame)
|
||||||
{
|
{
|
||||||
y_coord -= ROWS_PER_FRAME; // Subtract the ROWS_PER_FRAME from the pixel co-ord to get the panel co-ord.
|
y_coord -= ROWS_PER_FRAME; // Subtract the ROWS_PER_FRAME from the pixel co-ord to get the panel ROW (not really the 'y_coord' anymore)
|
||||||
paint_top_half = false;
|
painting_top_frame = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// We need to update the correct uint16_t in the rowBitStruct array, that gets sent out in parallel
|
||||||
|
int rowBitStruct_x_coord_uint16_t_position = (x_coord % 2) ? (x_coord-1):(x_coord+1);
|
||||||
|
|
||||||
for(int color_depth_idx=0; color_depth_idx<PIXEL_COLOR_DEPTH_BITS; color_depth_idx++) // color depth - 8 iterations
|
for(int color_depth_idx=0; color_depth_idx<PIXEL_COLOR_DEPTH_BITS; color_depth_idx++) // color depth - 8 iterations
|
||||||
{
|
{
|
||||||
uint16_t mask = (1 << color_depth_idx); // 24 bit color
|
int mask = (1 << color_depth_idx); // 24 bit color
|
||||||
|
|
||||||
// The destination for the pixel bitstream
|
// The destination for the pixel bitstream
|
||||||
//rowBitStruct *p = &matrix_framebuffer_malloc_1[back_buffer_id].rowdata[y_coord].rowbits[color_depth_idx]; //matrixUpdateFrames location to write to uint16_t's
|
//rowBitStruct *p = &matrix_framebuffer_malloc_1[back_buffer_id].rowdata[y_coord].rowbits[color_depth_idx]; //matrixUpdateFrames location to write to uint16_t's
|
||||||
rowBitStruct *p = &matrix_framebuffer_malloc_1[back_buffer_id].rowdata[y_coord].rowbits[color_depth_idx];
|
|
||||||
|
|
||||||
#ifdef SPLIT_MEMORY_MODE
|
// Find the memory address for the malloc for this framebuffer row.
|
||||||
if (y_coord >= SPLIT_MEMORY_ROWS_PER_FRAME ) {
|
rowColorDepthStruct *fb_row_malloc_ptr = (rowColorDepthStruct *) matrix_row_framebuffer_malloc[y_coord];
|
||||||
p = &matrix_framebuffer_malloc_2[back_buffer_id].rowdata[(y_coord-SPLIT_MEMORY_ROWS_PER_FRAME)].rowbits[color_depth_idx]; //matrixUpdateFrames location to write to uint16_t's
|
// Get the contents at this address, cast as a rowColorDepthStruct
|
||||||
|
rowBitStruct *p = &fb_row_malloc_ptr[back_buffer_id].rowbits[color_depth_idx]; //matrixUpdateFrames location to write to uint16_t's
|
||||||
#ifdef SERIAL_DEBUG
|
|
||||||
// Serial.printf("fb_malloc_2. y-coord: %d, calc. offset: %d \r\n", tmp_y_coord, (y_coord-SPLIT_MEMORY_ROWS_PER_FRAME) );
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
// int v = p->data[rowBitStruct_x_coord_uint16_t_position]; // persist what we already have
|
||||||
int v=0; // the output bitstream
|
int v=0; // the output bitstream
|
||||||
|
|
||||||
// if there is no latch to hold address, output ADDX lines directly to GPIO and latch data at end of cycle
|
// if there is no latch to hold address, output ADDX lines directly to GPIO and latch data at end of cycle
|
||||||
|
@ -545,17 +505,6 @@ void RGB64x32MatrixPanel_I2S_DMA::updateMatrixDMABuffer(int16_t x_coord, int16_t
|
||||||
if (gpioRowAddress & 0x08) v|=BIT_D; // 8
|
if (gpioRowAddress & 0x08) v|=BIT_D; // 8
|
||||||
if (gpioRowAddress & 0x10) v|=BIT_E; // 16
|
if (gpioRowAddress & 0x10) v|=BIT_E; // 16
|
||||||
|
|
||||||
/* ORIG
|
|
||||||
// need to disable OE after latch to hide row transition
|
|
||||||
if((x_coord) == 0) v|=BIT_OE;
|
|
||||||
|
|
||||||
// drive latch while shifting out last bit of RGB data
|
|
||||||
if((x_coord) == PIXELS_PER_LATCH-1) v|=BIT_LAT;
|
|
||||||
|
|
||||||
// need to turn off OE one clock before latch, otherwise can get ghosting
|
|
||||||
if((x_coord)==PIXELS_PER_LATCH-1) v|=BIT_OE;
|
|
||||||
*/
|
|
||||||
|
|
||||||
// need to disable OE after latch to hide row transition
|
// need to disable OE after latch to hide row transition
|
||||||
if((x_coord) == 0 ) v|=BIT_OE;
|
if((x_coord) == 0 ) v|=BIT_OE;
|
||||||
|
|
||||||
|
@ -564,9 +513,7 @@ void RGB64x32MatrixPanel_I2S_DMA::updateMatrixDMABuffer(int16_t x_coord, int16_t
|
||||||
|
|
||||||
// need to turn off OE one clock before latch, otherwise can get ghosting
|
// need to turn off OE one clock before latch, otherwise can get ghosting
|
||||||
if((x_coord)==PIXELS_PER_ROW-2) v|=BIT_OE;
|
if((x_coord)==PIXELS_PER_ROW-2) v|=BIT_OE;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
// turn off OE after brightness value is reached when displaying MSBs
|
// turn off OE after brightness value is reached when displaying MSBs
|
||||||
// MSBs always output normal brightness
|
// MSBs always output normal brightness
|
||||||
// LSB (!color_depth_idx) outputs normal brightness as MSB from previous row is being displayed
|
// LSB (!color_depth_idx) outputs normal brightness as MSB from previous row is being displayed
|
||||||
|
@ -579,25 +526,6 @@ void RGB64x32MatrixPanel_I2S_DMA::updateMatrixDMABuffer(int16_t x_coord, int16_t
|
||||||
if((x_coord) >= lsbBrightness) v|=BIT_OE; // For Brightness
|
if((x_coord) >= lsbBrightness) v|=BIT_OE; // For Brightness
|
||||||
}
|
}
|
||||||
|
|
||||||
/* When using the drawPixel, we are obviously only changing the value of one x,y position,
|
|
||||||
* however, the HUB75 is wired up such that it is always painting TWO lines at the same time
|
|
||||||
* and this reflects the parallel in-DMA-memory data structure of uint16_t's that are getting
|
|
||||||
* pumped out at high speed.
|
|
||||||
*
|
|
||||||
* So we need to ensure we persist the bits (8 of them) of the uint16_t for the row we aren't changing.
|
|
||||||
*
|
|
||||||
* The DMA buffer order has also been reversed (refer to the last code in this function)
|
|
||||||
* so we have to check for this and check the correct position of the MATRIX_DATA_STORAGE_TYPE
|
|
||||||
* data.
|
|
||||||
*/
|
|
||||||
int tmp_x_coord = x_coord;
|
|
||||||
if(x_coord%2)
|
|
||||||
{
|
|
||||||
tmp_x_coord -= 1;
|
|
||||||
} else {
|
|
||||||
tmp_x_coord += 1;
|
|
||||||
} // end reordering
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
// Development / testing code only.
|
// Development / testing code only.
|
||||||
Serial.printf("r value of %d, color depth: %d, mask: %d\r\n", red, color_depth_idx, mask);
|
Serial.printf("r value of %d, color depth: %d, mask: %d\r\n", red, color_depth_idx, mask);
|
||||||
|
@ -607,7 +535,7 @@ void RGB64x32MatrixPanel_I2S_DMA::updateMatrixDMABuffer(int16_t x_coord, int16_t
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|
||||||
if (paint_top_half)
|
if (painting_top_frame)
|
||||||
{ // Need to copy what the RGB status is for the bottom pixels
|
{ // Need to copy what the RGB status is for the bottom pixels
|
||||||
|
|
||||||
// Set the color of the pixel of interest
|
// Set the color of the pixel of interest
|
||||||
|
@ -616,13 +544,13 @@ void RGB64x32MatrixPanel_I2S_DMA::updateMatrixDMABuffer(int16_t x_coord, int16_t
|
||||||
if (red & mask) { v|=BIT_R1; }
|
if (red & mask) { v|=BIT_R1; }
|
||||||
|
|
||||||
// Persist what was painted to the other half of the frame equiv. pixel
|
// Persist what was painted to the other half of the frame equiv. pixel
|
||||||
if (p->data[tmp_x_coord] & BIT_R2)
|
if (p->data[rowBitStruct_x_coord_uint16_t_position] & BIT_R2)
|
||||||
v|=BIT_R2;
|
v|=BIT_R2;
|
||||||
|
|
||||||
if (p->data[tmp_x_coord] & BIT_G2)
|
if (p->data[rowBitStruct_x_coord_uint16_t_position] & BIT_G2)
|
||||||
v|=BIT_G2;
|
v|=BIT_G2;
|
||||||
|
|
||||||
if (p->data[tmp_x_coord] & BIT_B2)
|
if (p->data[rowBitStruct_x_coord_uint16_t_position] & BIT_B2)
|
||||||
v|=BIT_B2;
|
v|=BIT_B2;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
@ -633,25 +561,31 @@ void RGB64x32MatrixPanel_I2S_DMA::updateMatrixDMABuffer(int16_t x_coord, int16_t
|
||||||
if (green & mask) { v|=BIT_G2; }
|
if (green & mask) { v|=BIT_G2; }
|
||||||
if (blue & mask) { v|=BIT_B2; }
|
if (blue & mask) { v|=BIT_B2; }
|
||||||
|
|
||||||
// Copy
|
// Copy / persist
|
||||||
if (p->data[tmp_x_coord] & BIT_R1)
|
if (p->data[rowBitStruct_x_coord_uint16_t_position] & BIT_R1)
|
||||||
v|=BIT_R1;
|
v|=BIT_R1;
|
||||||
|
|
||||||
if (p->data[tmp_x_coord] & BIT_G1)
|
if (p->data[rowBitStruct_x_coord_uint16_t_position] & BIT_G1)
|
||||||
v|=BIT_G1;
|
v|=BIT_G1;
|
||||||
|
|
||||||
if (p->data[tmp_x_coord] & BIT_B1)
|
if (p->data[rowBitStruct_x_coord_uint16_t_position] & BIT_B1)
|
||||||
v|=BIT_B1;
|
v|=BIT_B1;
|
||||||
|
|
||||||
} // paint
|
} // paint
|
||||||
|
|
||||||
// 16 bit parallel mode
|
// 16 bit parallel mode
|
||||||
//Save the calculated value to the bitplane memory in reverse order to account for I2S Tx FIFO mode1 ordering
|
//Save the calculated value to the bitplane memory in reverse order to account for I2S Tx FIFO mode1 ordering
|
||||||
|
/*
|
||||||
if(x_coord%2){
|
if(x_coord%2){
|
||||||
p->data[(x_coord)-1] = v;
|
p->data[(x_coord)-1] = v;
|
||||||
} else {
|
} else {
|
||||||
p->data[(x_coord)+1] = v;
|
p->data[(x_coord)+1] = v;
|
||||||
} // end reordering
|
} // end reordering
|
||||||
|
*/
|
||||||
|
|
||||||
|
// 16 bit parallel mode
|
||||||
|
p->data[rowBitStruct_x_coord_uint16_t_position] = v;
|
||||||
|
|
||||||
|
|
||||||
} // color depth loop (8)
|
} // color depth loop (8)
|
||||||
|
|
||||||
|
@ -680,25 +614,11 @@ void RGB64x32MatrixPanel_I2S_DMA::updateMatrixDMABuffer(uint8_t red, uint8_t gre
|
||||||
uint16_t mask = (1 << color_depth_idx); // 24 bit color
|
uint16_t mask = (1 << color_depth_idx); // 24 bit color
|
||||||
|
|
||||||
// The destination for the pixel bitstream
|
// The destination for the pixel bitstream
|
||||||
rowBitStruct *p = &matrix_framebuffer_malloc_1[back_buffer_id].rowdata[matrix_frame_parallel_row].rowbits[color_depth_idx]; //matrixUpdateFrames location to write to uint16_t's
|
//rowBitStruct *p = &matrix_framebuffer_malloc_1[back_buffer_id].rowdata[matrix_frame_parallel_row].rowbits[color_depth_idx]; //matrixUpdateFrames location to write to uint16_t's
|
||||||
|
rowColorDepthStruct *fb_row_malloc_ptr = (rowColorDepthStruct *) matrix_row_framebuffer_malloc[matrix_frame_parallel_row];
|
||||||
#ifdef SPLIT_MEMORY_MODE
|
//Serial.printf("Accessing fb address: %d\r\n", fb_row_malloc_ptr);
|
||||||
if (matrix_frame_parallel_row >= SPLIT_MEMORY_ROWS_PER_FRAME ) {
|
|
||||||
p = &matrix_framebuffer_malloc_2[back_buffer_id].rowdata[(matrix_frame_parallel_row-SPLIT_MEMORY_ROWS_PER_FRAME)].rowbits[color_depth_idx]; //matrixUpdateFrames location to write to uint16_t's
|
|
||||||
|
|
||||||
#ifdef SERIAL_DEBUG
|
|
||||||
//Serial.printf("Using framebuffer_malloc_2. Row %d = Offset %d\r\n", matrix_frame_parallel_row, (matrix_frame_parallel_row-SPLIT_MEMORY_ROWS_PER_FRAME) );
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
#ifdef SERIAL_DEBUG
|
|
||||||
// Serial.printf("Using framebuffer_malloc_1. Row %d\r\n", matrix_frame_parallel_row );
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
|
|
||||||
|
rowBitStruct *p = &fb_row_malloc_ptr[back_buffer_id].rowbits[color_depth_idx]; //matrixUpdateFrames location to write to uint16_t's
|
||||||
|
|
||||||
for(int x_coord=0; x_coord < MATRIX_WIDTH; x_coord++) // row pixel width 64 iterations
|
for(int x_coord=0; x_coord < MATRIX_WIDTH; x_coord++) // row pixel width 64 iterations
|
||||||
{
|
{
|
||||||
|
|
|
@ -4,16 +4,9 @@
|
||||||
/***************************************************************************************/
|
/***************************************************************************************/
|
||||||
/* COMPILE-TIME OPTIONS - CONFIGURE AS DESIRED */
|
/* COMPILE-TIME OPTIONS - CONFIGURE AS DESIRED */
|
||||||
/***************************************************************************************/
|
/***************************************************************************************/
|
||||||
|
|
||||||
/* Enable serial debugging of the library, to see how memory is allocated etc. */
|
/* Enable serial debugging of the library, to see how memory is allocated etc. */
|
||||||
//#define SERIAL_DEBUG 1
|
//#define SERIAL_DEBUG 1
|
||||||
|
|
||||||
/* Split the framebuffer into two smaller memory allocations. Can allow for
|
|
||||||
* bigger resolution due to the fragmented memory map of the typical Arduino sketch
|
|
||||||
* even before setup(). Enabled by default.
|
|
||||||
*/
|
|
||||||
#define SPLIT_MEMORY_MODE 1
|
|
||||||
|
|
||||||
/* Use GFX_Root (https://github.com/mrfaptastic/GFX_Root) instead of
|
/* Use GFX_Root (https://github.com/mrfaptastic/GFX_Root) instead of
|
||||||
* Adafruit_GFX library. No real benefit unless you don't want Bus_IO & Wire.h library dependencies.
|
* Adafruit_GFX library. No real benefit unless you don't want Bus_IO & Wire.h library dependencies.
|
||||||
*/
|
*/
|
||||||
|
@ -145,20 +138,15 @@ struct rowColorDepthStruct {
|
||||||
};
|
};
|
||||||
|
|
||||||
/* frameStruct
|
/* frameStruct
|
||||||
* Note: This 'frameStruct' will contain ALL the data for a full-frame as BOTH 2x16-row frames are
|
* Note: A 'frameStruct' contains ALL the data for a full-frame (i.e. BOTH 2x16-row frames are
|
||||||
* are contained in parallel within the one uint16_t that is sent in parallel to the HUB75.
|
* are contained in parallel within the one uint16_t that is sent in parallel to the HUB75).
|
||||||
|
*
|
||||||
|
* This structure isn't actually allocated in one memory block anymore, as the library now allocates
|
||||||
|
* memory per row (per rowColorDepthStruct) instead.
|
||||||
*/
|
*/
|
||||||
#ifdef SPLIT_MEMORY_MODE
|
|
||||||
//#pragma message("Split DMA Memory Allocation Mode Enabled")
|
|
||||||
#define SPLIT_MEMORY_ROWS_PER_FRAME (ROWS_PER_FRAME/2)
|
|
||||||
struct frameStruct {
|
|
||||||
rowColorDepthStruct rowdata[SPLIT_MEMORY_ROWS_PER_FRAME];
|
|
||||||
};
|
|
||||||
#else
|
|
||||||
struct frameStruct {
|
struct frameStruct {
|
||||||
rowColorDepthStruct rowdata[ROWS_PER_FRAME];
|
rowColorDepthStruct rowdata[ROWS_PER_FRAME];
|
||||||
};
|
};
|
||||||
#endif
|
|
||||||
|
|
||||||
typedef struct rgb_24 {
|
typedef struct rgb_24 {
|
||||||
rgb_24() : rgb_24(0,0,0) {}
|
rgb_24() : rgb_24(0,0,0) {}
|
||||||
|
@ -323,13 +311,10 @@ class RGB64x32MatrixPanel_I2S_DMA : public Adafruit_GFX {
|
||||||
private:
|
private:
|
||||||
|
|
||||||
/* Pixel data is organized from LSB to MSB sequentially by row, from row 0 to row matrixHeight/matrixRowsInParallel
|
/* Pixel data is organized from LSB to MSB sequentially by row, from row 0 to row matrixHeight/matrixRowsInParallel
|
||||||
* (two rows of pixels are refreshed in parallel) */
|
* (two rows of pixels are refreshed in parallel)
|
||||||
frameStruct *matrix_framebuffer_malloc_1;
|
* Memory is allocated (malloc'd) by the row, and not in one massive chunk, for flexibility.
|
||||||
|
*/
|
||||||
#ifdef SPLIT_MEMORY_MODE
|
rowColorDepthStruct *matrix_row_framebuffer_malloc[ROWS_PER_FRAME];
|
||||||
/* In the case there's a 2nd non-contiguous block of DMA capable SRAM we can use, let us try and use that as well. */
|
|
||||||
frameStruct *matrix_framebuffer_malloc_2;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// ESP 32 DMA Linked List descriptor
|
// ESP 32 DMA Linked List descriptor
|
||||||
int desccount = 0;
|
int desccount = 0;
|
||||||
|
|
|
@ -1,21 +1,23 @@
|
||||||
// Example sketch which shows how to display a 64x32
|
// Example sketch which shows how to display a 64x32 animated GIF image stored in FLASH memory
|
||||||
// animated GIF image stored in FLASH memory
|
|
||||||
// on a 64x32 LED matrix
|
// on a 64x32 LED matrix
|
||||||
//
|
//
|
||||||
// To display a GIF from memory, a single callback function
|
|
||||||
// must be provided - GIFDRAW
|
|
||||||
// This function is called after each scan line is decoded
|
|
||||||
// and is passed the 8-bit pixels, RGB565 palette and info
|
|
||||||
// about how and where to display the line. The palette entries
|
|
||||||
// can be in little-endian or big-endian order; this is specified
|
|
||||||
// in the begin() method.
|
|
||||||
//
|
|
||||||
// The AnimatedGIF class doesn't allocate or free any memory, but the
|
|
||||||
// instance data occupies about 22.5K of RAM.
|
|
||||||
//
|
|
||||||
// Credits: https://github.com/bitbank2/AnimatedGIF/tree/master/examples/ESP32_LEDMatrix_I2S
|
// Credits: https://github.com/bitbank2/AnimatedGIF/tree/master/examples/ESP32_LEDMatrix_I2S
|
||||||
//
|
//
|
||||||
|
|
||||||
|
/* INSTRUCTIONS
|
||||||
|
*
|
||||||
|
* 1. First Run the 'ESP32 Sketch Data Upload Tool' in Arduino from the 'Tools' Menu.
|
||||||
|
* - If you don't know what this is or see it as an option, then read this:
|
||||||
|
* https://github.com/me-no-dev/arduino-esp32fs-plugin
|
||||||
|
* - This tool will upload the contents of the data/ directory in the sketch folder onto
|
||||||
|
* the ESP32 itself.
|
||||||
|
*
|
||||||
|
* 2. You can drop any animated GIF you want in there, but keep it to the resolution of the
|
||||||
|
* MATRIX you're displaying to. To resize a gif, use this online website: https://ezgif.com/
|
||||||
|
*
|
||||||
|
* 3. Have fun.
|
||||||
|
*/
|
||||||
|
|
||||||
#define FILESYSTEM SPIFFS
|
#define FILESYSTEM SPIFFS
|
||||||
#include <SPIFFS.h>
|
#include <SPIFFS.h>
|
||||||
#include <AnimatedGIF.h>
|
#include <AnimatedGIF.h>
|
||||||
|
|
Loading…
Reference in a new issue