【嵌入式】MCU(HC32F460)+SPI接口LCD液晶屏ILI9341 移植emWin记录2----移植emWin

本文主要是介绍【嵌入式】MCU(HC32F460)+SPI接口LCD液晶屏ILI9341 移植emWin记录2----移植emWin,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一 emWin移植准备

        完成了前一文点亮LCD屏之后,考虑到将要设计较为复杂的界面GUI,光用一些基本的绘图、显示字符接口不能满足要求,所以琢磨着再移植一套emWin,用来辅助设计GUI(由emWin的用户手册中可以看到,emWin是可以支持ILI9341的液晶屏驱动芯片的)。

        emWin的移植需要做一些准备:

        【1】首先,在自己的工程目录下建立GUI文件夹,以存放相关的文件,这边衍生出来的子文件夹可以起到归纳分类的作用:

         【2】其次,前往官网下载或者在本地keil路径下(不同人安装路径不一样,我本机的路径在此E:\Pro Software\Keil_v5\ARM\PACK\Keil\MDK-Middleware\7.8.0\emWin\)找到emWin文件,确认版本(我这边用的是emWin V5.42)。找到移植需要的文件,并移入自己工程目录下的GUI文件夹中,具体如下:

 (注:关于库文件的CM3不用太纠结,CM3与CM4的主要区别在于浮点数的处理,某些地方可以通用)

        【3】在keil中添加上面的源文件,并包含上面的头文件路径:

        完成了上述准备工作之后,即可正式进入移植。

二 emWin移植步骤

        【1】修改GUIConf.c文件。这边只要改一下GUI_NUMBYTES 的值,这个定义的是分配内存的大小,太大可能会使得GUI_Init()卡死,修改为合适的值即可:

#define GUI_NUMBYTES ((1024)*56)/*********************************************************************
*
*       Public code
*
**********************************************************************
*/
/*********************************************************************
*
*       GUI_X_Config
*
* Purpose:
*   Called during the initialization process in order to set up the
*   available memory for the GUI.
*/
void GUI_X_Config(void) {//// 32 bit aligned memory area//static U32 aMemory[GUI_NUMBYTES / 4];//// Assign memory to emWin//GUI_ALLOC_AssignMemory(aMemory, GUI_NUMBYTES);//// Set default font//GUI_SetDefaultFont(GUI_FONT_6X8);
}

        【2】修改GUIDRV_Template.c文件。这边需要修改画点函数_SetPixelIndex(),在其中嵌入

自己的打点函数LCD_DrawPoint(x, y),打点函数的具体实现在前一篇文章驱动点亮SPI屏中有记录:

/*********************************************************************
*
*       _SetPixelIndex
*
* Purpose:
*   Sets the index of the given pixel. The upper layers
*   calling this routine make sure that the coordinates are in range, so
*   that no check on the parameters needs to be performed.
*/
static void _SetPixelIndex(GUI_DEVICE * pDevice, int x, int y, LCD_PIXELINDEX PixelIndex) {#ifdef WIN32LCDSIM_SetPixelIndex(x, y, PixelIndex, pDevice->LayerIndex);#else//// Convert logical into physical coordinates (Dep. on LCDConf.h)//#if (LCD_MIRROR_X == 1) || (LCD_MIRROR_Y == 1) || (LCD_SWAP_XY == 1)int xPhys, yPhys;xPhys = LOG2PHYS_X(x, y);yPhys = LOG2PHYS_Y(x, y);#else#define xPhys x#define yPhys y#endifGUI_USE_PARA(pDevice);GUI_USE_PARA(x);GUI_USE_PARA(y);GUI_USE_PARA(PixelIndex);{//// Write into hardware ... Adapt to your system//// TBD by customer...//LCD_DrawPoint(x, y);  //重要:自行添加的画点函数}#if (LCD_MIRROR_X == 0) && (LCD_MIRROR_Y == 0) && (LCD_SWAP_XY == 0)#undef xPhys#undef yPhys#endif#endif
}

        【3】修改GUI_X_RTX.c文件。这个文件是官方已经适配好RTX操作系统的,同理还有GUI_X_uCOS、GUI_X_FreeRTOS等其他操作系统的适配,基本上拿过来就直接用了。主要作用就是用你的操作系统,建立一个时钟任务,来控制GUI的运行。

        【4】修改LCDConf.c文件。这边是最重要的一步,主要目的就是将emWin和LCD显示屏建立连接

        这边先贴出来代码,下面再一一说明:

#include "GUI.h"
#include "GUIDRV_FlexColor.h"extern void LCD_AllInit(void);
extern void LCD_ConfigureREG(void);
extern void LCD_WriteCMD(U8 Command);
extern void LCD_WriteDAT(U8 Command);
/*********************************************************************
*
*       Layer configuration (to be modified)
*
**********************************************************************
*/
//
// Physical display size
//   The display size should be adapted in order to match the size of
//   the target display.
//
#define XSIZE_PHYS 320
#define YSIZE_PHYS 240//
// Color conversion
//   The color conversion functions should be selected according to
//   the color mode of the target display. Detaileds can be found in
//   the chapter "Colors" in the emWin user manual.
//
//#define COLOR_CONVERSION GUICC_8888
#define COLOR_CONVERSION GUICC_565//
// Display driver
//   GUIDRV_WIN32 is for use only within the emWin Simulation
//   environment. In order to use the target display controller, the
//   according emWin display driver should be configured as it is
//   described in the chapter "Display Drivers" in the emWin user
//   manual. Beyond that sample configuration files can be found in
//   The folder "Sample\LCDConf\%DISPLAY_DRIVER%\".
//
//#define DISPLAY_DRIVER GUIDRV_WIN32
#define DISPLAY_DRIVER &GUIDRV_Template_API/*********************************************************************
*
*       Configuration checking
*
**********************************************************************
*/
#ifndef   XSIZE_PHYS#error Physical X size of display is not defined!
#endif
#ifndef   YSIZE_PHYS#error Physical Y size of display is not defined!
#endif
#ifndef   COLOR_CONVERSION#error Color conversion not defined!
#endif
#ifndef   DISPLAY_DRIVER#error No display driver defined!
#endif/*********************************************************************
*
*       Public code
*
**********************************************************************
*/
static void LcdWriteReg(U8 Data) {LCD_WriteCMD(Data);
}static void LcdWriteData(U8 Data) {LCD_WriteDAT(Data);
}static void LcdWriteDataMultiple(U8 * pData, int NumItems) {while(NumItems--){LCD_WriteDAT(*pData);}
}static void LcdReadDataMultiple(U8 * pData, int NumItems) {return ;
}/*********************************************************************
*
*       LCD_X_Config
*
* Function description
*   Called during the initialization process in order to set up the
*   display driver configuration.
*/
void LCD_X_Config(void) {
//    //1.这种初始化用不了颜色
//    GUI_DEVICE_CreateAndLink(DISPLAY_DRIVER, COLOR_CONVERSION, 0, 0);
//    if (LCD_GetSwapXY()) {
//    LCD_SetSizeEx (0, YSIZE_PHYS, XSIZE_PHYS);
//    LCD_SetVSizeEx(0, YSIZE_PHYS, XSIZE_PHYS);
//    } else {
//    LCD_SetSizeEx (0, XSIZE_PHYS, YSIZE_PHYS);
//    LCD_SetVSizeEx(0, XSIZE_PHYS, YSIZE_PHYS);
//    }//2.这种初始化可以使用颜色GUI_DEVICE * pDevice;GUI_PORT_API PortAPI = {0};pDevice = GUI_DEVICE_CreateAndLink(&GUIDRV_FlexColor_API, COLOR_CONVERSION, 0, 0);if (LCD_GetSwapXY()) {LCD_SetSizeEx (0, YSIZE_PHYS, XSIZE_PHYS);LCD_SetVSizeEx(0, YSIZE_PHYS, XSIZE_PHYS);} else {LCD_SetSizeEx (0, XSIZE_PHYS, YSIZE_PHYS);LCD_SetVSizeEx(0, XSIZE_PHYS, YSIZE_PHYS);}PortAPI.pfWrite8_A0  = LcdWriteReg;PortAPI.pfWrite8_A1  = LcdWriteData;PortAPI.pfWriteM8_A1 = LcdWriteDataMultiple;PortAPI.pfReadM8_A1  = LcdReadDataMultiple;GUIDRV_FlexColor_SetFunc(pDevice, &PortAPI, GUIDRV_FLEXCOLOR_F66709, GUIDRV_FLEXCOLOR_M16C0B8);
}/*********************************************************************
*
*       LCD_X_DisplayDriver
*
* Function description
*   Display driver callback function. This function is called by the
*   Display driver for certain purposes. Using GUIDRV_Win32 it is not
*   required to react to any command.
*/
int LCD_X_DisplayDriver(unsigned LayerIndex, unsigned Cmd, void * pData) {GUI_USE_PARA(LayerIndex);GUI_USE_PARA(Cmd);GUI_USE_PARA(pData);//    LCD_AllInit();//重要:添加初始化函数LCD_ConfigureREG();return 0;
}/*************************** End of file ****************************/

        (1)首先是定义LCD的屏幕尺寸、颜色格式以及驱动配置(这边定义的驱动GUIDRV_Template_API来自于上面第二步中的配置,因为不是触摸屏,这边只定义了打点函数,没有定义读点函数)

         (2)LCD_X_Config配置。这边提供两种连接驱动(GUI_DEVICE_CreateAndLink)的方法,一开始用方法一,连接的是上面定义好打点函数的驱动&GUIDRV_Template_API,测试emWin可以绘图,但是颜色只能显示黑白的。遂换用第二种方法,连接GUIDRV_FlexColor.h中定义的驱动&GUIDRV_FlexColor_API,写命令/数据接口全部都自己定义,具体的实现也可以参考上一篇文章驱动点亮SPI屏。

        这两种方法的区别,本质在于一个是直接把打点函数传递给&GUIDRV_FlexColor_API,另一个是把写命令/数据的接口传递给&GUIDRV_FlexColor_API,让其自己在内部实现打点。感觉两种方法本质上是一样的,导致方法一颜色用不了的区别可能在于下面GUIDRV_FlexColor_SetFunc接口的使用,这边我就没有继续细究了,如果有大佬看到的话,希望能够在评论区给我解个惑(20211108追更关于这个问题的解答:参考这篇帖子方法一:直接线性访问驱动和方法二:间接访问驱动)。

        (3)LCD_X_DisplayDriver配置。加入自己的LCD初始化函数。这边的LCD_ConfigureREG()同样参考上一篇文章,这边就不在赘述了:

        总上,配置完尺寸、颜色、驱动、初始化接口几项内容之后,emWin就实现与LCD的联系(LCDConf.c),再结合前面的自身内存的配置(GUIConf.c),以及与操作系统的关联(GUI_X_RTX.c),emWin就移植成功了。

三 emWin效果实现

        emWin移植成功之后,再结合LCD显示屏,就可以显示一些复杂的界面和内容了,这边测试仅试用一下emWin画⚪以及显示字符的功能(其中,字符显示部分,参考文章【嵌入式】emwin中插入中文字体——FontCvt的使用):

/**************************************************************************
* 函数名称: GUI_Task
* 功能描述: GUI子任务
* 输入参数: 
* 输出参数: 
* 返 回 值: 
* 其它说明: 
**************************************************************************/
void GUI_Task(void)
{//使用emwin来绘图LCD_InitGPIO();  //初始化几个GPIO口,包括BL、DC、RES以及SPI的CSLCD_InitSPI1();  //初始化SPI的几个口,包括SCK、MOSI以及MISOLCD_HardwareReset();  //LCD复位GUI_Init();  //emwin初始化LCD_DisplayDir(1);  //默认为横屏LCD_Clear(WHITE);  //清屏白色GUI_SetBkColor(GUI_WHITE);GUI_AA_SetFactor(4);GUI_EnableAlpha(1);GUI_UC_SetEncodeUTF8();GUI_SetFont(&GUI_FontFont32);GUI_SetColor(BLACK); GUI_SetTextMode(GUI_TM_TRANS);GUI_DispStringAt("Hello LCD!", 160, 120);GUI_DispStringAt("欢迎!", 160, 60);GUI_DrawCircle(80, 60, 40);while(1){os_dly_wait(1000);}
}

        

四 汇总几个遇到的坑

        (1)欢迎两个字明明加到字库中了,但是屏幕上始终显示不了

        【问题解决】:查阅资料发现问题在于,执行中文显示的语句GUI_DispStringAt("欢迎!", 160, 60)所在的源文件,没有转化为UTF-8编码格式,照这下面帖子的方法即可解决:emWin—显示汉字。

        (2)GUI内存配置的GUI_NUMBYTES不宜过大,一开始用的默认0x200000,程序挂死,后面改小为1024*56即可。

        (3)颜色显示仍然有问题,使用GUI自带的颜色库,例如GUI_GREEN等,实际显示的颜色与设置的不一致,具体原因没搞清楚,猜测可能跟LCD支持的颜色为GUICC_565(0xFFFF),而GUI库中的颜色为GUICC_888(0xFFFFFF)有关。

五 附录

        【1】【嵌入式】MCU(HC32F460)+SPI接口LCD液晶屏ILI9341 移植emWin记录1----点亮LCD屏

        【2】【嵌入式】emwin中插入中文字体——FontCvt的使用

        【3】HC32F460资料汇总

        【4】emWin V5.42中文说明书

这篇关于【嵌入式】MCU(HC32F460)+SPI接口LCD液晶屏ILI9341 移植emWin记录2----移植emWin的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/212608

相关文章

将sqlserver数据迁移到mysql的详细步骤记录

《将sqlserver数据迁移到mysql的详细步骤记录》:本文主要介绍将SQLServer数据迁移到MySQL的步骤,包括导出数据、转换数据格式和导入数据,通过示例和工具说明,帮助大家顺利完成... 目录前言一、导出SQL Server 数据二、转换数据格式为mysql兼容格式三、导入数据到MySQL数据

关于rpc长连接与短连接的思考记录

《关于rpc长连接与短连接的思考记录》文章总结了RPC项目中长连接和短连接的处理方式,包括RPC和HTTP的长连接与短连接的区别、TCP的保活机制、客户端与服务器的连接模式及其利弊分析,文章强调了在实... 目录rpc项目中的长连接与短连接的思考什么是rpc项目中的长连接和短连接与tcp和http的长连接短

Oracle查询优化之高效实现仅查询前10条记录的方法与实践

《Oracle查询优化之高效实现仅查询前10条记录的方法与实践》:本文主要介绍Oracle查询优化之高效实现仅查询前10条记录的相关资料,包括使用ROWNUM、ROW_NUMBER()函数、FET... 目录1. 使用 ROWNUM 查询2. 使用 ROW_NUMBER() 函数3. 使用 FETCH FI

详解Java如何向http/https接口发出请求

《详解Java如何向http/https接口发出请求》这篇文章主要为大家详细介绍了Java如何实现向http/https接口发出请求,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 用Java发送web请求所用到的包都在java.net下,在具体使用时可以用如下代码,你可以把它封装成一

Python MySQL如何通过Binlog获取变更记录恢复数据

《PythonMySQL如何通过Binlog获取变更记录恢复数据》本文介绍了如何使用Python和pymysqlreplication库通过MySQL的二进制日志(Binlog)获取数据库的变更记录... 目录python mysql通过Binlog获取变更记录恢复数据1.安装pymysqlreplicat

Java后端接口中提取请求头中的Cookie和Token的方法

《Java后端接口中提取请求头中的Cookie和Token的方法》在现代Web开发中,HTTP请求头(Header)是客户端与服务器之间传递信息的重要方式之一,本文将详细介绍如何在Java后端(以Sp... 目录引言1. 背景1.1 什么是 HTTP 请求头?1.2 为什么需要提取请求头?2. 使用 Spr

Servlet中配置和使用过滤器的步骤记录

《Servlet中配置和使用过滤器的步骤记录》:本文主要介绍在Servlet中配置和使用过滤器的方法,包括创建过滤器类、配置过滤器以及在Web应用中使用过滤器等步骤,文中通过代码介绍的非常详细,需... 目录创建过滤器类配置过滤器使用过滤器总结在Servlet中配置和使用过滤器主要包括创建过滤器类、配置过滤

正则表达式高级应用与性能优化记录

《正则表达式高级应用与性能优化记录》本文介绍了正则表达式的高级应用和性能优化技巧,包括文本拆分、合并、XML/HTML解析、数据分析、以及性能优化方法,通过这些技巧,可以更高效地利用正则表达式进行复杂... 目录第6章:正则表达式的高级应用6.1 模式匹配与文本处理6.1.1 文本拆分6.1.2 文本合并6

python与QT联合的详细步骤记录

《python与QT联合的详细步骤记录》:本文主要介绍python与QT联合的详细步骤,文章还展示了如何在Python中调用QT的.ui文件来实现GUI界面,并介绍了多窗口的应用,文中通过代码介绍... 目录一、文章简介二、安装pyqt5三、GUI页面设计四、python的使用python文件创建pytho

嵌入式QT开发:构建高效智能的嵌入式系统

摘要: 本文深入探讨了嵌入式 QT 相关的各个方面。从 QT 框架的基础架构和核心概念出发,详细阐述了其在嵌入式环境中的优势与特点。文中分析了嵌入式 QT 的开发环境搭建过程,包括交叉编译工具链的配置等关键步骤。进一步探讨了嵌入式 QT 的界面设计与开发,涵盖了从基本控件的使用到复杂界面布局的构建。同时也深入研究了信号与槽机制在嵌入式系统中的应用,以及嵌入式 QT 与硬件设备的交互,包括输入输出设