mirror of
https://github.com/espressif/esp-idf.git
synced 2024-10-05 20:47:46 -04:00
107 lines
3.6 KiB
C
107 lines
3.6 KiB
C
/*
|
|
* SPDX-FileCopyrightText: 2021 Espressif Systems (Shanghai) CO LTD
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
#include "freertos/FreeRTOS.h"
|
|
#include "soc/soc_caps.h"
|
|
#include "esp_lcd_common.h"
|
|
#if SOC_LCDCAM_SUPPORTED
|
|
#include "hal/lcd_ll.h"
|
|
#include "hal/lcd_hal.h"
|
|
|
|
typedef struct esp_lcd_platform_t {
|
|
portMUX_TYPE spinlock; // spinlock used to protect platform level resources
|
|
union {
|
|
void *panels[SOC_LCD_RGB_PANELS]; // array of RGB LCD panel instances
|
|
void *buses[SOC_LCD_I80_BUSES]; // array of i80 bus instances
|
|
}; // LCD peripheral can only work under either RGB mode or intel 8080 mode
|
|
} esp_lcd_platform_t;
|
|
|
|
esp_lcd_platform_t s_lcd_platform = {
|
|
.spinlock = (portMUX_TYPE)portMUX_INITIALIZER_UNLOCKED,
|
|
.buses = {} // initially the bus slots and panel slots are empty
|
|
};
|
|
|
|
int lcd_com_register_device(lcd_com_device_type_t device_type, void *device_obj)
|
|
{
|
|
int member_id = -1;
|
|
switch (device_type) {
|
|
case LCD_COM_DEVICE_TYPE_I80:
|
|
// search for a bus slot then register to platform
|
|
for (int i = 0; (i < SOC_LCD_I80_BUSES) && (member_id == -1); i++) {
|
|
portENTER_CRITICAL(&s_lcd_platform.spinlock);
|
|
if (!s_lcd_platform.buses[i]) {
|
|
s_lcd_platform.buses[i] = device_obj;
|
|
member_id = i;
|
|
}
|
|
portEXIT_CRITICAL(&s_lcd_platform.spinlock);
|
|
}
|
|
break;
|
|
case LCD_COM_DEVICE_TYPE_RGB:
|
|
// search for a panel slot then register to platform
|
|
for (int i = 0; (i < SOC_LCD_RGB_PANELS) && (member_id == -1); i++) {
|
|
portENTER_CRITICAL(&s_lcd_platform.spinlock);
|
|
if (!s_lcd_platform.panels[i]) {
|
|
s_lcd_platform.panels[i] = device_obj;
|
|
member_id = i;
|
|
}
|
|
portEXIT_CRITICAL(&s_lcd_platform.spinlock);
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
return member_id;
|
|
}
|
|
|
|
void lcd_com_remove_device(lcd_com_device_type_t device_type, int member_id)
|
|
{
|
|
switch (device_type) {
|
|
case LCD_COM_DEVICE_TYPE_I80:
|
|
portENTER_CRITICAL(&s_lcd_platform.spinlock);
|
|
if (s_lcd_platform.buses[member_id]) {
|
|
s_lcd_platform.buses[member_id] = NULL;
|
|
}
|
|
portEXIT_CRITICAL(&s_lcd_platform.spinlock);
|
|
break;
|
|
case LCD_COM_DEVICE_TYPE_RGB:
|
|
portENTER_CRITICAL(&s_lcd_platform.spinlock);
|
|
if (s_lcd_platform.panels[member_id]) {
|
|
s_lcd_platform.panels[member_id] = NULL;
|
|
}
|
|
portEXIT_CRITICAL(&s_lcd_platform.spinlock);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
#endif // SOC_LCDCAM_SUPPORTED
|
|
|
|
void lcd_com_mount_dma_data(dma_descriptor_t *desc_head, const void *buffer, size_t len)
|
|
{
|
|
size_t prepared_length = 0;
|
|
uint8_t *data = (uint8_t *)buffer;
|
|
dma_descriptor_t *desc = desc_head;
|
|
while (len > DMA_DESCRIPTOR_BUFFER_MAX_SIZE) {
|
|
desc->dw0.suc_eof = 0; // not the end of the transaction
|
|
desc->dw0.size = DMA_DESCRIPTOR_BUFFER_MAX_SIZE;
|
|
desc->dw0.length = DMA_DESCRIPTOR_BUFFER_MAX_SIZE;
|
|
desc->dw0.owner = DMA_DESCRIPTOR_BUFFER_OWNER_DMA;
|
|
desc->buffer = &data[prepared_length];
|
|
desc = desc->next; // move to next descriptor
|
|
prepared_length += DMA_DESCRIPTOR_BUFFER_MAX_SIZE;
|
|
len -= DMA_DESCRIPTOR_BUFFER_MAX_SIZE;
|
|
}
|
|
if (len) {
|
|
desc->dw0.suc_eof = 1; // end of the transaction
|
|
desc->dw0.size = len;
|
|
desc->dw0.length = len;
|
|
desc->dw0.owner = DMA_DESCRIPTOR_BUFFER_OWNER_DMA;
|
|
desc->buffer = &data[prepared_length];
|
|
desc = desc->next; // move to next descriptor
|
|
prepared_length += len;
|
|
}
|
|
}
|