基于Keil5移植LVGL,懂得原理之后什么开发板都可以移植

今天我们来移植一下LVGL,其实LVGL和Qt差不多,操作起来都很简单,看着官方文档都可以自己学习使用。

难就难在移植上面,移植个LVGL花了我三天才弄明白(虽然最后发现在一个很弱智的问题上耽误了我两天),网上的教程基本上只说应该怎么做,都没有说为什么要这么做,那么今天我们就来移植一下,知道原理之后就可以举一反三地去移植任何开发板了。

包括移植也是有官方手册的,英文好的小伙伴可以自行去查阅。官方手册讲的肯定比我好。

Porting — LVGL documentationicon-default.png?t=N7T8https://docs.lvgl.io/master/porting/index.html

虽然说是可以移植任何开发板,但是还是有要求的。

而我们常用的STM32C8T6是不符合要求的,GD32E230也是不符合要求的(虽然网上好像有移植成功的案例,但是作为初学者我们尽量还是拉高配置吧,省的不必要的麻烦),我这边用的是GD32F407VET6,配置上是足够的,包括ESP32-S3什么的也是满足要求的。

那么确认手上板子能够移植LVGL之后我们就可以进行下一步了,首先先要获取到LVGL的库文件,这个需要我们到github上面去下载。

https://github.com/lvgl/lvglicon-default.png?t=N7T8https://github.com/lvgl/lvgl考虑到github常常进不去,我把常用的版本给下载下来了。

链接:https://pan.baidu.com/s/1ubfUKtC1EuiZ_fM9rX4aHg?pwd=u5b7 
提取码:u5b7

里面是8.2,8.4和9.1。其中8.4和9.1是官方推荐版本,而8.2是部分网上视频教程里用的(某点原子),我接下来以8.4为例。

我们需要先整一个手上开发板可以用的工程文件,可以是空白的,但是要编译通过,这个不难,你就拿你之前项目的文件就行。

接着我们把下载好的LVGL解压。

得到的是这样的。

其中大部分文件是可以删除的,当然你不删也可以,但是会显得很臃肿。

我们来明确一下LVGL是什么东西,不要把LVGL当成是什么很高大上的东西,说白了就是第三方库,和我们自己封装的头文件什么的是一个样的,只不过人家LVGL的文件数量多了点,仅此而已。

既然和我们写的头文件一样,那么我们只需要保留.c和.h文件即可,其他后缀的都是可以删除了。

接下来我们就对解压过后的LVGL进行删减。

只留下面这三个文件夹和两个.h文件即可。

为什么把其他的删掉呢?那当然是因为它们不包含我们需要的.c和.h文件啦。

顺便把留下的这五个东西介绍一下。

第一个demos,跟它的名字一样,存放的是官方的demo,因此实际上我们也是可以删掉的,留下来的原因就是等我们移植完LVGL之后可以调用一下demo看看效果。

第二个examples,存放的是一些关于示例代码,我们只需要留下里面的porting即可。

然后porting里的文件是三个(一对的.c和.h算一个)。

disp是显示相关的模板,fs是文件管理的模板,indev是输入相关的模板。

因为我的屏幕没有输入,也不需要文件管理,因此对于我来说只需要disp即可,但是其他没用到的最好也别删了,保不齐什么时候就用上了(没事的时候打算整个触摸屏再移植一下LVGL玩玩)。

然后就是src,这个就是核心的文件了,一个都不能少。

lv_conf_tempplate.h这个文件是配置文件。

lvgl.h这个头文件里面包含了我们常用的其他头文件,因此我们只需要包含这个就等同于包含了其他很多头文件了,比较方便。

然后配置文件的名字我们需要把后面的_template给删掉,变成lv_conf.h。

修改名称的原因就在于LVGL里面源文件中包含的配置文件的名称就是lv_conf.h,

那么接下来我们就要把LVGL的文件加入到我们的工程文件里了。

接下来在你工程文件里存放模块的文件夹里新建一个文件夹来存放LVGL的文件(名字随意,我这边就叫LVGL)

接下来在这个LVGL的文件夹里再新建两个文件夹,名字随意,一个用来存放源码一个用来存放官方demo,其实是可以省略的,只不过我们要运行官方demo的话区分一下会比较清晰,如果不需要运行官方demo的可以省略这一步,直接跳到下一步。

我这边就借用一下某点原子的命名规则了,建立两个文件夹,GUI用来存放源码,GUI_APP存放官方demo。

接着在把删减过后的LVGL中的demos文件夹放到GUI_APP里(跳过上面一步的小伙伴这一步也可以省略)。然后在GUI里建一个名为lvgl的文件(这里名字是固定的,一定要这么写,否则后续要修改的地方非常多),接着把剩余两个文件夹和两个头文件放到这个lvgl文件夹里。

最后就是这样子的。

其他命名啊,结构啊什么的都无所谓,但是lvgl这个文件夹是一定要安装这个名字去创建的,并且要像上图这样把删减后的LVGL放进去。

我们随便打开一个lvgl源码的头文件,可以发现人家包含头文件是有一定结构的,因此我们需要按照这种结构去存放lvgl的源码。

那么接下来我们的工程里有了LVGL的代码了,我们就需要再Keil里设置包含住他们。

首先现在分组里添加上一些组。

可以看到我设置的分组的名字和我们LVGL文件名称是基本对应的,不过实际上分组名字是可以随便起的,你要是乐意起个ABCD都是可以的,但是名字对应上方便我们查看代码。

这边分组完我们还需要把每个文件夹内的.c文件放进每个分组里,记得不要重复包含,也不要漏掉包含,就这一步是最容易疏漏的。

每个文件夹内的文件夹里的.c文件也是需要包含的,虽然大部分我们用不上,但是作为小白的我们暂时还分不出哪些是可以不用的,所以我们就麻烦一下全部加上。

记住,是对应分组的文件夹里的所有.c文件!!!

这一步是最繁琐也是最容易出问题的,如果移植失败,那么90%的原因是在这里!!!

添加完毕之后在左侧我们就可以看到添加进工程的文件了。

但是光添加进来没有用,我们还得让编译器知道这些文件的存在,因此我们还需要添加头文件路径。就像下图这样,可以顺便把C99给勾上。

一共往上面添加四个路径。

这个根据大家命名的不同需要选择不同的路径,但是通过上面这个图,大体上可以看出需要包含哪些路径的对吧。

上面步骤做完之后我们就可以先编译一下看看有没有什么不对的地方了。

会有警告,但是正常情况不会有报错,我们选择无视警告。

如果有报错的,一般是报这个错误Error: L6218E: Undefined symbol ,这种情况基本就是给分组添加文件的时候没有添加全,换句话说就是没把某些LVGL的文件添加进我们的工程中。

报错的小伙伴重新再把文件添加进工程里,记得不要重复包含。

只有警告的小伙伴可以跟我一起进行下一步了。

我们再新建个分组,把配置文件添加进来,不进行这一步也可以,但是就需要在Keil外面修改配置文件了。

第一步我们就先把配置文件给打开,打开的方式就是把这个判断条件修改为1。

包括其他配置也可以在这个文件里进行修改。

比如下面这个宏定义,就是给LVGL分配的内存大小。

默认给48KB,我的板子是给的起,但是有些板子内存资源可能有些不太够,分太多会编译报错,那么就可以在这边修改小一点,但是最小不要小于2KB。

还有颜色相关的配置也在这边。

默认的颜色像素是RGB565,如果你们的屏幕不是RGB565的话,就是在这边修改的。

下一步把显示相关的.c和.h也打开,也是通过修改判断条件。

接着修改宏定义来指定屏幕大小。

按照顺序分别是宽和高。

如果是8.2版本的是没有这俩宏定义的,需要自己去定义。

接着我们需要指定LVGL的缓冲模式,有三种,按照顺序效果是越来越好的,但是相对的内存消耗会更大,大家量力而行,一般来说模式一都够用了。

选择的方式就是把另外两种模式的代码注释掉。

看的出来我们上面的宏定义是有用在这边的。

然后因为这是和显示相关的,因此我们肯定是需要自己的屏幕驱动代码的,不懂写的小伙伴可以回顾一下我之前的文章。

【硬件模块】ST7735S(1.8寸TFT-LCD)-CSDN博客SPI,英文全称Serial Peripheral Interface,即串行外围设备接口,是一种高速、全双工、同步的串行通信总线。我们之前说过I2C,那么我们就拿I2C和SPI做个对比。SPI和I2C对比,优势在于SPI的传输速率比I2C快得多,劣势在于SPI需要用的通信线比较多。SCK(Serial Clock):串行时钟线,由主设备产生,用于同步数据传输。MOSI(Master Output Slave Input):主机输出从机输入线,主设备通过这条线发送数据给从设备。https://blog.csdn.net/m0_63235356/article/details/139422591?spm=1001.2014.3001.5502

在LVGL显示相关的文件中包含我们的屏幕驱动代码。

接着在disp_init这个函数里调用我们屏幕的初始化函数。

disp_flush函数里调用我们的打点函数。

其中函数的形参area是个结构体,一共有四个成员,分别是x1,x2,y1,y2,表示打点的范围,color_p是像素点的数组,我们一般是要强转一下类型再给我们的打点函数。

上面是我写的打点函数,大家根据自己的驱动去填写参数即可。

那么关于显示的配置我们就算配置完了,因为我的屏幕不是触摸的,因此我就不配置输入相关的了,不过大致流程也是这样的。

接下来我们回到我们的主函数。

先包含两个头文件,我这是只有输出的,如果配置输入的话,还需要包含另外一个"lv_port_indev_template.h"。

#include "lvgl.h"
#include "lv_port_disp_template.h"

我们最后还需要给LVGL一个心跳,也就是我们需要使用定时器,在一定的时间里调用一个LVGL的函数lv_tick_inc。

不懂定时器的小伙伴可以参考我之前的文章。

STM32,ESP32,GD32都有,往前翻一翻。

然后我们每隔1ms进入一次中断,我们就给lv_tick_inc这个函数传入1,如果是隔5ms一次进入中断,那么给的参数就是5,也就是说隔几毫秒,传的参数就是几。

接着在主循环里每隔5ms调用一次lv_timer_handler就可以了。

那么我们就算是移植完LVGL了。

但是我们要看效果,还需要调用一些初始化函数,下面是输出的初始化,如果有输入的话,还需哟啊调用输入初始化函数。

    lv_init();lv_port_disp_init();

初始化完之后我们再创建组件,鉴于大家都是小白(包括我),因此我们直接从官方文档里复制示例代码来看看效果。

我这里就只贴出main.c的内容,大家大概知道流程是怎么样的就可以了。

#include "board.h"
#include "Z_TFT.h"
#include "Z_Timer.h"#include "lvgl.h"
#include "lv_port_disp_template.h"void TIMER0_UP_TIMER9_IRQHandler(void){if(timer_interrupt_flag_get(TIMER9, TIMER_INT_FLAG_UP) == SET)    {        timer_interrupt_flag_clear(TIMER9, TIMER_INT_FLAG_UP); // 清除中断标志位   lv_tick_inc(1);}
}int main(void){board_init();Z_Timer_Init();lv_init();lv_port_disp_init();lv_obj_t * bar1 = lv_bar_create(lv_scr_act());lv_obj_set_size(bar1, 100, 20);lv_obj_center(bar1);lv_bar_set_value(bar1, 70, LV_ANIM_OFF);while (1){delay_ms(5);lv_timer_handler();}
}

效果是这样的。

那么至此,各位就成功地移植LVGL了,完整做下来之后实际上没有那么困难,LVGL就是普普通通的第三方库,我们要做的就是把这一堆文件塞进我们工程里,再让我们的编译器识别到这些文件就行了。

最后再总结一下可能会出错的点。

第一个点就将文件包含进工程里,这是最容易出错的地方,多了少了都可能会出错。

第二就是配置显示的时候自己的屏幕驱动了,不过跟我的驱动走没什么大问题。

我这边留个课后作业,我们做删减的时候不是把LVGL的官方demo留下来了嘛,给大家的作业就是运行这些官方demo,流程是一样的,就是把文件包含进工程里,再添加头文件路径,然后改个配置调用即可。

算了,带大家做一遍吧。

一样是先创建分组,再把文件弄进来,板子配置一般的小伙伴运行stress那个demo,配置好的可以用music。我这边用stress,本来想玩一玩那个music的,但是调了一会还是运行不了music然后就放弃了。

接着是头文件路径,我们一样是加进来,不同的demo用不同的路径。

然后在配置文件里打开stress的宏定义,不同的demo打开不同的宏定义。

完成上面步骤之后就可以运行了,在main.c里包含一下lv_demo_stress.h,再调用lv_demo_stress就可以了。

我们就可以看见屏幕正在飞快刷新着,这是LVGL在不断地调用各种组件。

至此,我相信各位小伙伴都已经会移植LVGL了。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/341797.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

AI大模型时代必须关注的数据库 DuckDB1.0 正式发布

开源数据库DuckDB1.0 经过内部6年的打磨,积累了30万行代码,1.8万star,2024.06.03号正式发布了1.0版本(代号 Snow Duck)。 我们新一代程序员,没能见证MySQL 1.0、PostgreSQL 1.0、Windows 1.0、Linux 1.0、…

HTML跳动的爱心

目录 写在前面 HTML简介 程序设计 修改文字 推荐系列 写在后面 写在前面 本期小编给大家分享可以写字的html动态爱心代码,一起来看看叭~ HTML简介 HTML(HyperText Markup Language)是一种用于创建网页的标记语言。它是互联网的基础&…

Etcd Raft架构设计和源码剖析1:宏观架构

Etcd Raft架构设计和源码剖析1:宏观架构 | Go语言充电站 序言 Etcd提供了一个样例contrib/raftexample,用来展示如何使用etcd raft。这篇文章通过raftexample介绍如何使用etcd raft。 raft服务 raftexample是一个分布式KV数据库,客户端可…

三十六、openlayers官网示例Earthquake Clusters解析——在聚合图层鼠标触摸显示五角星

官网demo地址: Earthquake Clusters 这篇展示了鼠标触摸聚合图层点位显示五角星的效果。 首先是初始化地图,加载了一个KML格式的矢量数据源,extractStyles为false表示不从kml数据源中提取样式。使用Select添加了鼠标选中的交互事件 vector …

《微服务大揭秘:SpringBoot与SpringCloud的魔法组合》

加入我们的探险队伍,一起深入SpringBoot与SpringCloud构建的微服务世界。以轻松幽默的笔触,带你一步步揭开微服务架构的神秘面纱,从服务发现的智能地图Eureka,到API网关Zuul的城市门卫,每一个环节都充满了惊喜。不仅如…

htb_solarlab

端口扫描 80,445 子域名扫描 木有 尝试使用smbclient连接445端口 Documents目录可查看 将Documents底下的文件下载到本地看看 xlsx文件里有一大串用户信息,包括username和password 先弄下来 不知道在哪登录,也没有子域名,于是返回进行全端…

chat4-Server端保存聊天消息到mysql

本文档描述了Server端接收到Client的消息并转发给所有客户端或私发给某个客户端 同时将聊天消息保存到mysql 服务端为当前客户端创建一个线程,此线程接收当前客户端的消息并转发给所有客户端或私发给某个客户端同时将聊天消息保存到mysql 本文档主要总结了将聊天…

UnityAPI学习之游戏物体的方法使用

目录 游戏物体 创建游戏物体的三种方式 组建的获取和查找 游戏物体的方法与其他成员变量 游戏物体的生成 游戏物体的激活状态/标签(tag)/层级(layer) 游戏物体的激活与失活 游戏物体的查找 1. 名称查找(Find) 2. 通过标签查找游戏物体(FindGameObjectWithT…

v1.2.70-FastJson的AutoType机制研究

v1.2.70-FastJson的AutoType机制研究 最近在对接Alexa亚马逊语音技能,Smart Home Skill Apis时,有一个配置的JSON字符串是这样的: { "capabilityResources": {"friendlyNames": [{"type": "asset",…

json和axion结合

目录 java中使用JSON对象 在pom.xml中导入依赖 使用 public static String toJSONString(Object object)把自定义对象变成JSON对象 json和axios综合案例 使用的过滤器 前端代码 响应和请求都是普通字符串 和 请求时普通字符串,响应是json字符串 响应的数据是…

使用 Django 连接 MySQL 数据库

文章目录 步骤一:安装必要的库和驱动步骤二:配置数据库连接步骤三:执行数据库迁移步骤四:开始使用 MySQL 数据库创建一个模型迁移模型到数据库使用模型进行数据操作创建新记录:查询记录:更新记录&#xff1…

基于百度接口的实时流式语音识别系统

目录 基于百度接口的实时流式语音识别系统 1. 简介 2. 需求分析 3. 系统架构 4. 模块设计 4.1 音频输入模块 4.2 WebSocket通信模块 4.3 音频处理模块 4.4 结果处理模块 5. 接口设计 5.1 WebSocket接口 5.2 音频输入接口 6. 流程图 程序说明文档 1. 安装依赖 2.…

Element ui图片上传

前言 对于广大小白来说,图片上传简直是上传难,难于上青天!废话不多说,步入正题,您就瞧好吧! 步骤一:前端使用element ui组件(upload上传) 我个人喜欢使用第二个组件&a…

运放应用1 - 反相放大电路

1.前置知识 反相放大电路存在 负反馈电路 ,工作在线性区,可以利用 虚短 概念来分析电路。 注:运放的 虚断 特性是一直存在的,虚短特性则需要运放工作在 线性区 有关运放的基础知识,可以参考我的另外一篇文章&#xff…

BGP基础配置

BGP 邻居关系建立,与路由条目宣告是分开的配置的 1)直连的 EBGP 邻居关系建立 [r1]bgp1 启动 BGP 协议,启动时需要键入 AS号;没有多进程概念 [r1-bgp]router-id 1.1.1.1 建议配置 RID;若不配置将自动生成一规则同 OSPF 相同 [r1-bgp]pe…

门面模式Api网关(SpringCloudGateway)

1. 前言 当前通过Eureka、Nacos解决了服务注册和服务发现问题,使用Spring Cloud LoadBalance解决了负载均衡的需求,同时借助OpenFeign实现了远程调用。然而,现有的微服务接口都直接对外暴露,容易被外部访问。为保障对外服务的安全…

【kubernetes】探索k8s集群的pod控制器详解(Deployment、StatefulSet、DaemonSet、Job、CronJob)

目录 一、Pod控制器及其功用 二、pod控制器有多种类型 2.1ReplicaSet 2.1.1ReplicaSet主要三个组件组成 2.2Deployment 2.3DaemonSet 2.4StatefulSet 2.5Job 2.6Cronjob 三、Pod与控制器之间的关系 3.1Deployment 3.2SatefulSet 3.2.1StatefulSet三个组件 3.2.2为…

引擎:UI

一、控件介绍 Button 按钮 创建一个按钮 按钮禁用 精灵模式 颜色模式 缩放模式 绑定点击事件 EditBox 输入框 Layout 布局 支持水平排列、垂直排列、背包排列 PageView 页面视图 ProgressBar 进度条 RichText 富文本 绑定点击事件 事件可以被其它标签包裹 图文混排 Scroll…

Redis位图

简介 在我们平时开发过程中,会有一些bool型数据需要存取,比如用户一年的签到记录,签了是1,没签是0,要记录365天。如果使用普通的key/value,每个用户要记录365个,当用户上亿的时候,需…

本地知识库开源框架Fastgpt、MaxKB产品体验

本地知识库开源框架Fastgpt、MaxKB产品体验 背景fastgpt简介知识库共享部署 MaxKB总结 背景 上一篇体验了Quivr、QAnything两个开源知识库模型框架,这次介绍两款小众但是体验比较好的产品。 fastgpt 简介 FastGPT 是一个基于 LLM 大语言模型的知识库问答系统&am…