ESP32 ESP-IDF LVGL移植和Wokwi仿真

news/2025/1/16 7:42:29/

陈拓 2023/10/21-2023/10/25

1. ESP-IDF开发环境

ESP-IDF的LVGL移植包括2个组件:

  • lvgl
  • lvgl_esp32_drivers

目前lvgl_esp32_drivers在ESP-IDF 5.0以上版本编译通不过,所以我们安装ESP-IDF 4.4.5。

从https://dl.espressif.cn/dl/esp-idf/下载

安装说明见《Windows系统安装ESP32 ESP-IDF开发环境》

https://blog.csdn.net/chentuo2000/article/details/133922505?spm=1001.2014.3001.5501

2. ESP-IDF lvgl和lvgl_esp32_drivers组件下载

  1. LVGL相关文档

https://docs.lvgl.io/8.3/

  • lvgl和lvgl_esp32_drivers下载网址

在下面这个网址可以找到很多ESP-IDF组件,因为是国内网络下载比较容易。

https://components.espressif.com/

LVGL各个版本的组件也可以在这里找到,目前最新版本是8.3.10。

LVGL也可以从github下载。

下载LVGL之后还要从github下载lvgl_esp32_drivers。

下面我们从github下载lvgl和lvgl_esp32_drivers。

  • 下载lvgl

lvgl和lvgl_esp32_drivers组件可以下载到ESP-IDF的components目录下面,也可以下载到项目的components下面。

下面我们将两个组件下载到ESP-IDF的components目录下面。

进入components:

cd components

下载LVGL最新版本

https://github.com/lvgl/lvgl/tags

下载v8.3.10:

git clone -b v8.3.10 https://github.com/lvgl/lvgl.git

如果下载失败多选择几遍。

提示:

如果访问github超时或者下载速度慢,可以试试用Watt Toolkit加速,网址:

https://steampp.net/

Watt Tookit可以从Microsoft Store直接安装:

  • 下载lvgl_esp32_drivers

git clone https://github.com/lvgl/lvgl_esp32_drivers.git

3. 创建新项目

  • 在F:盘上建一个工作目录esp

  • 进入esp

  • 在esp下再建一个项目目录esp32_lvgl

mkdir esp32_lvgl

  • 复制测试项目到工作目录

xcopy D:\Espressif\frameworks\esp-idf-v4.4.5\examples\get-started\sample_project\ F:\esp\esp32_lvgl\ /E

  • cd esp32_lvgl

4. 编译项目

4.1 编译LVGL

  • 设定目标芯片

idf.py set-target esp32

  • 配置项目

idf.py menuconfig

1) 将闪存设置为4MB

2) 选择显示屏控制芯片ILI9341

3) 定义引脚

接线表(对照硬件电路):

ILI9341             ESP32

GND                   GND

VCC                    VIN

SCK                    D18

MOSI                  D23

RES                    D4

D/C                     D2

CS                      D15

LED                    空

MISO                  空

4) 无背光控制

5) 屏幕方向 - 横屏

6) 交换16位颜色的2个字节

ILI9341的颜色深度是16位,选择Color depth. (16:RGB565),颜色值用2字节表示,ESP32是小端Little Endian模式,先发送低位字节,如果颜色失真,可以选交换颜色值的高低字节。

保存,退出。

  • 在lvgl_esp32_drivers\lvgl_helpers.h中添加宏定义
/**********************      DEFINES*********************/#define LV_HOR_RES_MAX 320
#define LV_VER_RES_MAX 240
#define SPI_HOST_MAX 3
  • 编译LVGL

idf.py build

看到这些提示LVGL编译就成功了。

4.2 编译项目

  • 修改main.c

现在的main.c是空的:

/**********************      DEFINES*********************/#define LV_HOR_RES_MAX 320
#define LV_VER_RES_MAX 240
#define SPI_HOST_MAX 3

用下面的代码替换:

#include <stdio.h>
#include "freertos/FreeRTOS.h"
#include "freertos/event_groups.h"
#include "lvgl.h"
#include "freertos/semphr.h"
#include "esp_system.h"
#include "lvgl_helpers.h"/**********************      DEFINES*********************/
#define LV_TICK_PERIOD_MS 1/***********************  STATIC PROTOTYPES**********************/
static void lv_tick_task(void *arg);
static void guiTask(void *pvParameter);
static void create_demo_application(void);/***********************   APPLICATION MAIN**********************/
void app_main(void)
{// xTaskCreate(guiTask, "gui", 4096*2, NULL, 1, NULL);// xTaskCreatePinnedToCore(guiTask, "gui", 4096*2, NULL, 1, NULL, 1);/* NOTE: When not using Wi-Fi nor Bluetooth you can pin the guiTask to core 0 */xTaskCreatePinnedToCore(guiTask, "gui", 4096*2, NULL, 0, NULL, 1);
}/* Creates a semaphore to handle concurrent call to lvgl stuff* If you wish to call *any* lvgl function from other threads/tasks* you should lock on the very same semaphore! */
SemaphoreHandle_t xGuiSemaphore;static void guiTask(void *pvParameter)
{(void) pvParameter;xGuiSemaphore = xSemaphoreCreateMutex();lv_init();/* Initialize SPI or I2C bus used by the drivers */lvgl_driver_init();lv_color_t *buf1 = heap_caps_malloc(DISP_BUF_SIZE * sizeof(lv_color_t), MALLOC_CAP_DMA);assert(buf1 != NULL);/* Use double buffered when not working with monochrome displays */lv_color_t* buf2 = heap_caps_malloc(DISP_BUF_SIZE * sizeof(lv_color_t), MALLOC_CAP_DMA);assert(buf2 != NULL);static lv_disp_draw_buf_t disp_buf;uint32_t size_in_px = DISP_BUF_SIZE;/* Initialize the working buffer depending on the selected display. */lv_disp_draw_buf_init(&disp_buf, buf1, buf2, size_in_px);lv_disp_drv_t disp_drv;lv_disp_drv_init(&disp_drv);disp_drv.hor_res = LV_HOR_RES_MAX;disp_drv.ver_res = LV_VER_RES_MAX;disp_drv.flush_cb = disp_driver_flush;disp_drv.draw_buf = &disp_buf;lv_disp_drv_register(&disp_drv);/* Create and start a periodic timer interrupt to call lv_tick_inc */const esp_timer_create_args_t periodic_timer_args = {.callback = &lv_tick_task,.name = "periodic_gui"};esp_timer_handle_t periodic_timer;ESP_ERROR_CHECK(esp_timer_create(&periodic_timer_args, &periodic_timer));ESP_ERROR_CHECK(esp_timer_start_periodic(periodic_timer, LV_TICK_PERIOD_MS * 1000));/* Create the demo application */create_demo_application();while (1){/* Delay 1 tick (assumes FreeRTOS tick is 10ms */vTaskDelay(pdMS_TO_TICKS(10));lv_task_handler();/* Try to take the semaphore, call lvgl related function on success */if (pdTRUE == xSemaphoreTake(xGuiSemaphore, portMAX_DELAY)) {lv_task_handler();xSemaphoreGive(xGuiSemaphore);}}
}static void create_demo_application(void)
{/* Get the current screen  */lv_obj_t * scr = lv_disp_get_scr_act(NULL);    /*Create a Label on the currently active screen*/lv_obj_t * label1 =  lv_label_create(scr);/*Modify the Label's text*/lv_label_set_text(label1, "Hello\nworld");/* Align the Label to the center* NULL means align on parent (which is the screen now)* 0, 0 at the end means an x, y offset after alignment*/lv_obj_align(label1, LV_ALIGN_CENTER, 0, 0);
}static void lv_tick_task(void *arg) {(void) arg;lv_tick_inc(LV_TICK_PERIOD_MS);
}
  • 编译项目

idf.py build

在build目录下生成引导加载程序bootloader.bin、应用程序main.bin和分区表partition-table.bin三个ESP32运行所需的bin文件。

5. Wokwi仿真

详细说明见《用Wokwi仿真ESP-IDF项目》

https://blog.csdn.net/chentuo2000/article/details/133963728?spm=1001.2014.3001.5501

5.1 硬件模拟电路

  • 打开Wokwi网页

https://wokwi.com/

选择ESP32。

  • 选择项目

打开ILI9341 TFT LCD显示屏项目。

  • Wokwi编程、仿真界面

左边是arduino的程序代码编写区。右边是电路连接和仿真区。

5.2 仿真

  • 在代码编辑器中按F1

选择Upload Firmware and Start Simulation…

  • 选择main.bin文件

  • Wokwi模拟器显示

6. demos仿真

  • 选择Show some widget和Enable slide show

  • 修改main.c

添加头文件:

#include "demos/lv_demos.h"
  • 修改create_demo_application函数
static void create_demo_application(void)
{#if defined CONFIG_LV_USE_DEMO_WIDGETSlv_demo_widgets();#elif defined CONFIG_LV_USE_DEMO_KEYPAD_AND_ENCODERlv_demo_keypad_encoder();#elif defined CONFIG_LV_USE_DEMO_BENCHMARKlv_demo_benchmark();#elif defined CONFIG_LV_USE_DEMO_STRESSlv_demo_stress();#else#error "No demo application selected."#endif
}
  • 编译项目

idf.py build

内存不足,无法运行lv_demo_widget。请将LV_MEM_SIZE设置为至少38KB(38ul*1024ul)。建议使用48KB。

再编译成功。

  • 仿真

参考文档

  1.  ESP32 ESP-IDF LVGL8.3.3移植
    https://blog.csdn.net/chentuo2000/article/details/128269394?spm=1002014.3001.5501


http://www.ppmy.cn/news/1178677.html

相关文章

Docker 容器服务的注册、发现及Docker安全

目录 Docker容器服务的注册和发现 1、什么是服务注册与发现&#xff1f; 2、什么是consul consul的部署 1、环境准备 2、部署consul服务器 1&#xff09;建立 Consul 服务 2&#xff09;设置代理&#xff0c;在后台启动 consul 服务端 3&#xff09;查看集群信息 4&a…

sheng的学习笔记-【中】【吴恩达课后测验】Course 3 - 结构化机器学习项目 - 第一周测验

课程3_第1周_测验题 目录&#xff1a;目录 要解决的问题 ① 这个例子来源于实际项目&#xff0c;但是为了保护机密性&#xff0c;我们会对细节进行保护。 ② 现在你是和平之城的著名研究员&#xff0c;和平之城的人有一个共同的特点&#xff1a;他们害怕鸟类。 ③ 为了保护…

海外广告投放必看,如何使用Quora广告开拓新流量市场?

虽然在Quora 上学习广告相对容易&#xff0c;但需要大量的试验和错误才能找出最有效的方法。一些广告技巧可以让您的工作更有效率。这篇文章将介绍如何有效进行quora广告投放与有价值的 Quora 广告要点&#xff0c;这将为您节省数万美元的广告支出和工作时间&#xff01;往下看…

ubuntu安装idea

idea下载 选择第一个.tar.gz(Linux)下载 解压后 进入bin目录在终端运行./idea.sh即可

代码审计及示例

简介&#xff1a; 代码安全测试是从安全的角度对代码进行的安全测试评估。 结合丰富的安全知识、编程经验、测试技术&#xff0c;利用静态分析和人工审核的方法寻找代码在架构和编码上的安全缺陷&#xff0c;在代码形成软件产品前将业务软件的安全风险降到最低。 方法&#x…

使用Python和WebDriver自动化处理网页弹窗!

引言&#xff1a; 在进行Web自动化测试时&#xff0c;经常会遇到网页上的弹窗&#xff0c;这些弹窗可能是提示框、确认框、输入框等。对于这些弹窗的处理&#xff0c;我们可以使用Python和WebDriver进行自动化操作&#xff0c;以提高测试效率。本文将介绍如何使用Python和WebD…

前端(二十五)——前端实现 OCR 图文识别的详细步骤与示例代码

&#x1f601;博主&#xff1a;小猫娃来啦 &#x1f601;文章核心&#xff1a;前端实现 OCR 图文识别的详细步骤与示例代码 文章目录 简介确定使用的 OCR API创建前端界面添加图像上传功能发送识别请求和处理识别结果完善代码添加注释结论附录 简介 在现代应用程序中&#xff…

Linux--安装与配置虚拟机及虚拟机服务器坏境配置与连接---超详细教学

一&#xff0c;操作系统介绍 1.1.什么是操作系统 操作系统&#xff08;Operating System&#xff0c;简称OS&#xff09;是一种系统软件&#xff0c;它是计算机硬件和应用软件之间的桥梁。它管理计算机的硬件和软件资源&#xff0c;为应用程序提供接口和服务&#xff0c;并协调…