LVGL开发教程-FreeRTOS中的LVGL

系列文章目录

知不足而奋进 望远山而前行


目录

系列文章目录

文章目录

前言

重要步骤

lv_tick_inc(x)

lv_timer_handler()

1. 声明一把锁

2. 初始化这把锁

3. 创建一个任务

4. 编写任务的内容

完整示例代码

总结


前言

在嵌入式系统开发中,使用LVGL(LittlevGL)库可以轻松实现现代化的图形用户界面(GUI)。移植LVGL到特定的硬件平台是一项关键任务,需要遵循一系列重要步骤以确保库能够正确运行并充分发挥其功能。本文将介绍在LVGL移植过程中的关键步骤,帮助开发者快速上手并成功将LVGL集成到自定义的嵌入式项目中。


重要步骤

在LVGL移植文档Set up a project — LVGL documentation中, 总共有5个重要步骤:

  1. 调用lv_init() :lvgl的初始化核心
  2. 初始化我们自己的屏幕驱动相关,I2C,SPI,ST7789,CST816T等
  3. 调用lv_port_disp_init()(负责显示部分) 和 lv_port_indev_init()(负责输入部分)
  4. 调用lv_tick_inc(x) :这个是负责维持lvgl核心库的心跳
  5. 调用lv_timer_handler(): 负责lvgl中的相关任务

前面1,2,3个步骤都是与初始化相关,我们把它们写在启动任务中

static void start_task() {
    taskENTER_CRITICAL();
		
    lv_init();
    lv_port_disp_init();
    lv_port_indev_init();

    vTaskDelete(start_handler);

    taskEXIT_CRITICAL();
}

lv_tick_inc(x)

我们需要维持lvgl自身的心跳,以便让它能够处理它自身内部的任务,例如动画的渲染,长按单击事件的判断等等, 我们将这个函数放到FreeRTOS的调用函数中

例如,我们在main.c文件中定义如下名称函数:

void vApplicationTickHook( void ){
	lv_tick_inc(1);
}

注意,定义了这个函数,我们还需要打开它对应的宏,在FreeRTOSConfig.h文件中第57行

#define configUSE_TICK_HOOK	1

lv_timer_handler()

lvgl中的所有的任务处理都在这个函数里,lvgl是线程不安全的,这个函数同时只能有一个线程调用它,保险起见我们需要给它的调用加一把锁

1. 声明一把锁

SemaphoreHandle_t 	xMutex;

2. 初始化这把锁

xMutex = xSemaphoreCreateMutex();

3. 创建一个任务

xTaskCreate(lvgl_timer_task, "lvgl_timer_task", 1024, NULL, 4, NULL);

4. 编写任务的内容

void lvgl_timer_task(void * pvParameters)
{
	xMutex = xSemaphoreCreateMutex();
	
	lv_obj_t * btn2 = lv_btn_create(lv_scr_act());
	lv_obj_set_size(btn2,100,100);
	lv_obj_align(btn2, LV_ALIGN_CENTER, 0, 0);
	lv_obj_add_flag(btn2, LV_OBJ_FLAG_CHECKABLE);
	
	lv_obj_t * label2 = lv_label_create(btn2);
	lv_label_set_text(label2, "Toggle");
	lv_obj_center(label2);
	
	while(1)
	{
		if(pdTRUE == xSemaphoreTake(xMutex,portMAX_DELAY))	
		{
			lv_timer_handler();		
			xSemaphoreGive(xMutex);
		}
		vTaskDelay(pdMS_TO_TICKS(1));			
	}
}

创建刷新的任务

static void lvgl_refresh_task() {
		lv_obj_t * label = lv_label_create(lv_scr_act());
		lv_label_set_text(label, "count:");	
		
		lv_obj_set_style_text_color(label, lv_palette_main(LV_PALETTE_BLUE), 0);
		lv_obj_set_pos(label,10,10);
		int count = 0;
        while(1) {
    			
    				lv_label_set_text_fmt(label,"count:%d",count);
    				count++;
            vTaskDelay(pdMS_TO_TICKS(1000));
    				printf("task1 \r\n");
        }
}

完整示例代码

#include "gd32f4xx.h"
#include "systick.h"
#include <stdio.h>
#include <string.h>
#include "main.h"

#include "bsp_usart.h"

#include "FreeRTOS.h"
#include "task.h"
#include "lvgl.h"
#include "lv_port_disp.h"
#include "lv_port_indev.h"
#include "i2c.h"
#include "st7789.h"
#include "semphr.h"

static TaskHandle_t           start_handler;
static SemaphoreHandle_t 	xMutex;


void Usart0_recv(uint8_t *data, uint32_t len) {
	printf("Usart0_recv:%s \r\n",data);
}

static void lvgl_refresh_task() {
		lv_obj_t * label = lv_label_create(lv_scr_act());
		lv_label_set_text(label, "count:");	
		
		lv_obj_set_style_text_color(label, lv_palette_main(LV_PALETTE_BLUE), 0);
		lv_obj_set_pos(label,10,10);
		int count = 0;
    while(1) {
			
				lv_label_set_text_fmt(label,"count:%d",count);
				count++;
        vTaskDelay(pdMS_TO_TICKS(1000));
				printf("task1 \r\n");
    }
}


void lvgl_timer_task(void * pvParameters)
{
	xMutex = xSemaphoreCreateMutex();
	
	lv_obj_t * btn2 = lv_btn_create(lv_scr_act());
	lv_obj_set_size(btn2,100,100);
	lv_obj_align(btn2, LV_ALIGN_CENTER, 0, 0);
	lv_obj_add_flag(btn2, LV_OBJ_FLAG_CHECKABLE);
	
	lv_obj_t * label2 = lv_label_create(btn2);
	lv_label_set_text(label2, "Toggle");
	lv_obj_center(label2);
	
	printf("lcd_refresh_tas\r\n");
	
	while(1)
	{
		if(pdTRUE == xSemaphoreTake(xMutex,portMAX_DELAY))	
		{
			lv_timer_handler();		
			xSemaphoreGive(xMutex);
		}
		vTaskDelay(pdMS_TO_TICKS(1));			
	}
}


static void start_task() {
    taskENTER_CRITICAL();
		
		bsp_usart_dma_init(115200);
		printf("start123\r\n");
	
		I2C_init();
		
		lv_init();
		lv_port_disp_init();
		lv_port_indev_init();
    
		xTaskCreate(lvgl_timer_task, "lvgl_timer_task", 1024, NULL, 4, NULL);
		xTaskCreate(lvgl_refresh_task, "lvgl_refresh_task", 256, NULL, 4, NULL);
		
    vTaskDelete(start_handler);

    taskEXIT_CRITICAL();
}

int main(void)
{
    nvic_priority_group_set(NVIC_PRIGROUP_PRE4_SUB0);
    systick_config();
	
    xTaskCreate(start_task, "start_task", 512, NULL, 1, &start_handler);
    vTaskStartScheduler();

    while(1) {
    }
}


void vApplicationTickHook( void ){
	lv_tick_inc(1);
}

总结

LVGL的移植过程涉及几个关键步骤,包括初始化LVGL核心、配置屏幕和输入设备驱动、以及确保LVGL正常运行所需的心跳维持和任务处理。通过调用lv_init()初始化核心、配置驱动、设置心跳维持以及处理定时器任务等步骤,开发者可以在不同的硬件平台上成功移植LVGL库,从而为项目提供强大的GUI功能支持。这些步骤不仅确保了LVGL在目标设备上的稳定运行,还为开发者提供了自定义GUI界面的基础。

  • 15
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
FreeRTOS是一个开源的实时操作系统内核,被广泛应用于嵌入式系统。ESP32是一款具有双核处理器和Wi-Fi功能的芯片,通过使用ESP-IDF开发框架可以进行软件开发。在ESP32-IDF开发,使用FreeRTOS的消息队列可以实现不同任务之间的通信。 在ESP32开发,可以通过断服务程序(Interrupt Service Routine,ISR)来发送消息到消息队列,并在任务通过接收方法响应。 首先,我们需要创建一个全局的消息队列句柄,可以使用xQueueCreate函数来创建一个消息队列。例如,可以使用以下代码创建一个大小为10的消息队列: xQueueHandle messageQueue = xQueueCreate(10, sizeof(int)); 然后,在断服务程序,可以使用xQueueSendFromISR方法将消息发送到消息队列。例如,可以使用以下代码将一个整数值发送到消息队列: int value = 100; xQueueSendFromISR(messageQueue, &value, NULL); 在任务,可以使用xQueueReceive方法从消息队列接收消息并进行响应。例如,可以使用以下代码从消息队列接收一个整数值并打印出来: int receivedValue; xQueueReceive(messageQueue, &receivedValue, portMAX_DELAY); printf("Received value: %d\n", receivedValue); 需要注意的是,在接收消息时,可以通过指定第三个参数来设置等待时间。例如,使用portMAX_DELAY表示无限等待,即直到接收到消息为止。 通过以上步骤,我们可以实现在ESP32开发使用FreeRTOS消息队列进行断服务消息发送与响应。这种方式可以实现不同任务之间的通信和同步,提高系统的并发性和实时性。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

薛慕昭

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值