嵌入式图形用户界面uc/gui在nios II上的移植_nios gui-程序员宅基地

技术标签: include  manager  buffer  存储  Linux Kernel  嵌入式  图形  

 

uc/gui是一个优秀的嵌入式图形用户界面,这几天的工作就是将它移植到nios II系统上。前人也做了一些工作,不过大部分都是针对其他硬核处理器,针对nios II软核处理器的移植资料那简直是凤毛麟角。在阅读了相关文档后,我决定自己亲自动手实践,这下面的很多过程都是自己摸索出来的,并通过了实验的验证。这只是一个初步的移植,也许在以后的更复杂的应用中,还需要对其进行调整。但对目前我的应用而言,应该足够了。

写这篇文章的目的一是由于自己记性不好,所以需要给自己留个备忘,免得以后忘的一干二净;二是给有需要的朋友提供一些参考,也好相互交流,共同进步。请大家多提宝贵意见。

一、源码和文档下载

http://www.ucgui.com/上有很多不同版本的源码下载,目前能下到的最新版本是3.98,不过还有一些组件不是很完整,但作基础开发已经够用了。

ucgui3.98源码下载地址:uC-GUI-V3-98.zip

ucgui最新版用户手册下载地址:uC-GUI-user.rar

开发软件:quartus II 6.0, Nios II IDE 6.0

二、移植过程

先来看看解压后都有些什么东西:

 

如图,核心的东西包括ConfigGUI两个文件夹,这里面是ucgui的所有源码和配置文件。ConvertColor包含彩色转换函数,ConvertMono包含灰度到彩色转换的函数,Core包含核心程序,Font是字体文件,LCDDriver包含多种控制器驱动,Widget是窗口控件库,WM是窗口库,提供复杂的功能。其他文件夹包含一些应用范例以及一些有用的工具,留待慢慢探索。

1config文件的移植:

Config文件夹是ucgui的配置文件夹,里面有3个文件:

GUIConf.hgui的基本属性配置文件,有很多开关可以配置,具体可以参考ucgui的用户手册,这里只需配置几个必要的参数如下:

#ifndef GUICONF_H

#define GUICONF_H

#define GUI_OS                    (1)  /* 支持操作系统,nios系统自带了ucosII,所以我们选择此项,使gui支持该操作系统 */

#define GUI_SUPPORT_TOUCH         (0)  /* 支持触摸屏,由于暂时没有用触摸屏,所以关掉这个开关 */

#define GUI_SUPPORT_MOUSE         (0)  /* 支持鼠标,暂时关闭 */

#define GUI_SUPPORT_UNICODE       (1)  /* Unicode字符串支持 */

#define GUI_DEFAULT_FONT          &GUI_Font6x8/* 默认字体 */

#define GUI_ALLOC_SIZE            12500/* WMmemery device分配的内存 */

#define GUI_WINSUPPORT            1  /* Window manager available */

#define GUI_SUPPORT_MEMDEV        0  /* Memory devices available,由于下载到的源代码中缺少memery device组件的源码,所以关闭此项 */

#define GUI_SUPPORT_AA            1  /* Anti aliasing available */

#endif  /* Avoid multiple inclusion */

LCDConf.hLCD控制器的硬件配置文件,这个文件与硬件直接相关,一般是根据你所使用的LCD的类型和所用的LCD控制器的类型来配置。我的配置是一块640*480TFT LCD,支持18位色,不过我只使用16位,RGB565色彩模式,足矣。LCD控制器就是自己写的一个硬件模块,挂在avalon总线上,负责读取显示缓冲区中的数据,然后按照该LCD的时序输出显示到LCD上。显示缓冲区直接开辟在系统内存中,系统使用一块SDRAM作为系统内存,CPU可以直接对其进行32位读写访问。通过仔细阅读ucgui的用户手册,可以知道,在我这种硬件配置条件下,可以选择LCDLin32.c这个驱动文件(后面将详细讲述对LCDLin32.c的修改与移植),那么对应了LCD_CONTROLLER 必须配置为3200

#ifndef LCDCONF_H

#define LCDCONF_H

#define LCD_XSIZE      (640)  /* X-resolution of LCD, Logical coor. */

#define LCD_YSIZE      (480)  /* Y-resolution of LCD, Logical coor. */

#define LCD_BITSPERPIXEL (16) /* 每个象素点需要的Bit */

#define LCD_CONTROLLER 3200   /* 控制器名称 */

#define LCD_ENDIAN_BIG      0 /* 选择little endian */

#define LCD_FIXEDPALETTE  565 /* 选择RGB565色彩模式 */

#define LCD_SWAP_RB         1 /* gui默认为GRB565,定义这个开关可以使之转换为RGB565,即交换RB */

//#define LCD_VRAM_ADR        0x20000000 /*  显示缓冲区起始地址,这个宏定义定义了显示缓冲区的地址,在驱动文件LCDLin32.c中要用到这个地址。由于我的显示缓冲区要在程序运行起来之后,由malloc函数在系统内存中分配,所以这个地址无法预先定义。于是我取消了这个宏定义,而改为一个指针变量,直接放在LCDLin32.c中,通过修改其中的一些函数定义,完全可以实现这样的改变。 */

//#define LCD_READ_MEM(Off)        IORD_32DIRECT((U32 *)LCD_VRAM_ADR + ((U32)Off) << 2),0)

//#define LCD_WRITE_MEM(Off,data)  IOWR_32DIRECT((U32 *)LCD_VRAM_ADR + ((U32)Off) << 2),0,Data)

/* 使用驱动LCDLin32.c需要定义的函数,用于读写显示缓冲区,由于把显示缓冲区的地址指针放到LCDLin32.c中去了,所以这两个函数也直接放到LCDLin32.c中去,在这里就可以不要了 */

#define LCD_INIT_CONTROLLER()  LCD_Controller_Init() /* LCD 控制器初始化函数,由gui_init()调用,我在这里替换为自己的控制器初始化函数LCD_Controller_Init(),该函数在后文中叙述 */

#endif /* LCDCONF_H */

GUITouchConf.h:触摸屏的配置文件,暂时没有使用。

至此,config文件移植完毕。

2LCD device驱动的移植

这里所说的LCD device驱动移植主要是指前面所说的LCDLin32.c文件的修改。LCDLin32.cGUI/LCDDriver文件夹中,其中定义了几个关键的函数,用于gui对显示缓冲区进行操作,如

void LCD_L0_SetPixelIndex(int x, int y, int PixelIndex)/* 画点 */

unsigned int LCD_L0_GetPixelIndex(int x, int y)/* 读点 */

void LCD_L0_XorPixel(int x, int y)/* 异或点 */

void LCD_L0_DrawHLine(int x0, int y,  int x1) /* 画水平线 */

void LCD_L0_DrawVLine(int x, int y0,  int y1) /* 画垂直线 */

void LCD_L0_FillRect(int x0, int y0, int x1, int y1) /* 矩阵填充 */

void LCD_L0_DrawBitmap(int x0, int y0,

                       int xsize, int ysize,

                       int BitsPerPixel,

                       int BytesPerLine,

                       const U8 GUI_UNI_PTR * pData, int Diff,

                       const LCD_PIXELINDEX* pTrans) /* 画位图 */

void LCD_On (void) /* 打开LCD */

void LCD_Off (void) /* 关闭LCD */

由于我的LCD不具备打开和关闭功能,所以LCD_On()LCD_Off()定义为空函数。在上述的几个画点画线函数中,与硬件(显示缓冲区)直接相关的就是

LCD_WRITE_MEM(Off,data)/*写存储器*/

LCD_READ_MEM           /*读存储器*/

我们只需要将这两个宏定义修改一下,使之指向我们自己定义的操作。由于这两个函数是对显示缓冲区进行读写操作,所以需要先知道显示缓冲区的起始地址,在我的系统中该缓冲区由malloc函数得到,于是需要预先定义一个指针变量,用来存储显示缓冲区的首地址:

U32      *lcd_framebuffer0;      /* our frame buffer first address */

但是预编译无法处理malloc函数,所以我将这个操作放到自定义的LCD_Controller_Init()函数中执行。这里使用了Nios系统中hal库的一些函数,所以需要先包含相应的头文件:

#include <stdio.h>

#include <stdlib.h>

#include <unistd.h>

#include <string.h>

#include <alt_types.h>

#include "io.h"

#include "sys/alt_alarm.h"

#include "sys/alt_cache.h"

#include "system.h"

#include "priv/alt_file.h"

这其中实际上有几个头文件是没有必要包含进来的,不过我也没去管它,所以就都写在这里了,应该不会有什么害处。

#define LCD_BYTESPERFRAME  LCD_XSIZE * LCD_YSIZE * LCD_BITSPERPIXEL / 8

void LCD_Controller_Init(void)

{

    lcd_framebuffer0 = (U32 *)alt_uncached_malloc(LCD_BYTESPERFRAME);

    memset( (void *)lcd_framebuffer0, 0x0, LCD_BYTESPERFRAME ) ; /* reset the frame buffer to 0x0 */

    IOWR_32DIRECT( VGA_CONTROLLER_0_BASE, 0, 0x0 ); /* Reset the VGA controller */

    IOWR_32DIRECT( VGA_CONTROLLER_0_BASE, 4, lcd_framebuffer0 ); /* Where our frame buffer starts */

    IOWR_32DIRECT( VGA_CONTROLLER_0_BASE, 8, LCD_BYTESPERFRAME ); /* amount of memory needed */  

    IOWR_32DIRECT( VGA_CONTROLLER_0_BASE, 0, 0x1 ); /* Set the go bit. */

}

这个函数初始化了显示缓冲区,并对LCD控制器进行配置,使之正确运行。其中,VGA_CONTROLLER_0_BASELCD控制器的基地址,由system.h文件定义,所以要将它include进来。LCD_BYTESPERFRAME为每帧需要的字节数。由宏定义得到。这样,在系统调用GUI_Init()函数时,LCD_Controller_Init()函数也会被调用,这就可以保证gui在做任何操作以前,显示缓冲区都已经准备好了,LCD控制器也已经配置好并已经开始运行了。

实际上,LCDLin32.c文件中对LCD_READ_MEMLCD_WRITE_MEM已经做了定义,有了lcd_framebuffer0这个指针变量之后,我们只需要对LCD_READ_MEMLCD_WRITE_MEM定义做一些修改,使之指向lcd_framebuffer0所指的缓冲区即可:

#define   LCD_READ_MEM(Off)        (*((U32 *)lcd_framebuffer0 + ((U32)Off)))

#define   LCD_WRITE_MEM(Off, Data) *((U32 *)lcd_framebuffer0 + ((U32)Off)) = Data

这样,LCDLin32.c就基本上修改完毕了。当然,我们还可以修改其画点,画线,矩阵填充,画位图等函数,使之对于我们特定的硬件更加优化,以提高执行效率,这是后话。到目前为止,gui已经能够正确地操作我们的硬件了。

三、运行第一个程序:hello_gui

下面,我们就让刚移植好的gui到实际的系统上去运行一下。

1、配置好FPGA的硬件;

2、打开nios II IDE,以hello_world工程为模版建立一个新的工程hello_gui

3、将ucguiConfigGUI两个文件夹(包含有我们刚刚修改过的几个文件)复制到工程目录下;

4、在hello_gui工程选项中添加如下include paths

yourprojectdir/software/hello_gui/Config

yourprojectdir/software/hello_gui/GUI/Core

yourprojectdir/software/hello_gui/GUI/Widget

yourprojectdir/software/hello_gui/GUI/WM

5、修改hello_world.c的内容为:

#include "GUI.H"

void main(void) {

GUI_Init();                 /* 初始化GUI,同时初始化LCD控制器和显示缓冲区 */

GUI_SetBkColor(GUI_BLUE);   /* 设置背景色为蓝色*/

GUI_Clear();                /* 清屏为背景色 */

GUI_SetColor(GUI_RED);      /* 设置前景色为红色(后面画图操作将用该前景色) */

GUI_DispString("Hello world!"); /* 显示hello world! */

while(1);

}

6、在system library选项中选择RTOSMicroC/OS-II

这时如果编译工程,会出现一些未定义错误,类似于GUI_X_未定义等等,原来是缺少GUI_X.c文件,这个文件定义了guirtos的接口,以及debug错误报告等函数,于是我们需要添加这个文件。经过寻找,在Sample文件夹中发现了GUI_X这个文件夹,打开一看,里面有我们所需要的GUI_X.c文件,由于我们使用了MicroC/OS-II操作系统,所以我们使用其中的GUI_X_uCOS.c文件,将这个文件copy到工程目录下,并在文件结尾添加下面这几行:

void GUI_X_Log     (const char *s) { GUI_USE_PARA(s); }

void GUI_X_Warn    (const char *s) { GUI_USE_PARA(s); }

void GUI_X_ErrorOut(const char *s) { GUI_USE_PARA(s); }

并将其中的:

void GUI_X_ExecIdle (void)

{

    OS_X_Delay(1);

}

改为:

void GUI_X_ExecIdle (void)

{

    OSTimeDly(1);       /* 调用uCOS-II的延时程序 */

}

方可编译通过。

7Debug as hardware,这样,就可以看见结果了。我的结果是LCD上用篮底红字显示出了“hello world!”字样。表示ucgui3.98Nios II上初步移植成功!

四、总结

至此,ucguiNios II上的移植获得了初步的成功。当然,这只是一小步,ucgui中还有丰富的功能,这些都将在以后的实践中慢慢摸索。

 

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/itismine/article/details/4145280

智能推荐

c# 调用c++ lib静态库_c#调用lib-程序员宅基地

文章浏览阅读2w次,点赞7次,收藏51次。四个步骤1.创建C++ Win32项目动态库dll 2.在Win32项目动态库中添加 外部依赖项 lib头文件和lib库3.导出C接口4.c#调用c++动态库开始你的表演...①创建一个空白的解决方案,在解决方案中添加 Visual C++ , Win32 项目空白解决方案的创建:添加Visual C++ , Win32 项目这......_c#调用lib

deepin/ubuntu安装苹方字体-程序员宅基地

文章浏览阅读4.6k次。苹方字体是苹果系统上的黑体,挺好看的。注重颜值的网站都会使用,例如知乎:font-family: -apple-system, BlinkMacSystemFont, Helvetica Neue, PingFang SC, Microsoft YaHei, Source Han Sans SC, Noto Sans CJK SC, W..._ubuntu pingfang

html表单常见操作汇总_html表单的处理程序有那些-程序员宅基地

文章浏览阅读159次。表单表单概述表单标签表单域按钮控件demo表单标签表单标签基本语法结构<form action="处理数据程序的url地址“ method=”get|post“ name="表单名称”></form><!--action,当提交表单时,向何处发送表单中的数据,地址可以是相对地址也可以是绝对地址--><!--method将表单中的数据传送给服务器处理,get方式直接显示在url地址中,数据可以被缓存,且长度有限制;而post方式数据隐藏传输,_html表单的处理程序有那些

PHP设置谷歌验证器(Google Authenticator)实现操作二步验证_php otp 验证器-程序员宅基地

文章浏览阅读1.2k次。使用说明:开启Google的登陆二步验证(即Google Authenticator服务)后用户登陆时需要输入额外由手机客户端生成的一次性密码。实现Google Authenticator功能需要服务器端和客户端的支持。服务器端负责密钥的生成、验证一次性密码是否正确。客户端记录密钥后生成一次性密码。下载谷歌验证类库文件放到项目合适位置(我这边放在项目Vender下面)https://github.com/PHPGangsta/GoogleAuthenticatorPHP代码示例://引入谷_php otp 验证器

【Python】matplotlib.plot画图横坐标混乱及间隔处理_matplotlib更改横轴间距-程序员宅基地

文章浏览阅读4.3k次,点赞5次,收藏11次。matplotlib.plot画图横坐标混乱及间隔处理_matplotlib更改横轴间距

docker — 容器存储_docker 保存容器-程序员宅基地

文章浏览阅读2.2k次。①Storage driver 处理各镜像层及容器层的处理细节,实现了多层数据的堆叠,为用户 提供了多层数据合并后的统一视图②所有 Storage driver 都使用可堆叠图像层和写时复制(CoW)策略③docker info 命令可查看当系统上的 storage driver主要用于测试目的,不建议用于生成环境。_docker 保存容器

随便推点

网络拓扑结构_网络拓扑csdn-程序员宅基地

文章浏览阅读834次,点赞27次,收藏13次。网络拓扑结构是指计算机网络中各组件(如计算机、服务器、打印机、路由器、交换机等设备)及其连接线路在物理布局或逻辑构型上的排列形式。这种布局不仅描述了设备间的实际物理连接方式,也决定了数据在网络中流动的路径和方式。不同的网络拓扑结构影响着网络的性能、可靠性、可扩展性及管理维护的难易程度。_网络拓扑csdn

JS重写Date函数,兼容IOS系统_date.prototype 将所有 ios-程序员宅基地

文章浏览阅读1.8k次,点赞5次,收藏8次。IOS系统Date的坑要创建一个指定时间的new Date对象时,通常的做法是:new Date("2020-09-21 11:11:00")这行代码在 PC 端和安卓端都是正常的,而在 iOS 端则会提示 Invalid Date 无效日期。在IOS年月日中间的横岗许换成斜杠,也就是new Date("2020/09/21 11:11:00")通常为了兼容IOS的这个坑,需要做一些额外的特殊处理,笔者在开发的时候经常会忘了兼容IOS系统。所以就想试着重写Date函数,一劳永逸,避免每次ne_date.prototype 将所有 ios

如何将EXCEL表导入plsql数据库中-程序员宅基地

文章浏览阅读5.3k次。方法一:用PLSQL Developer工具。 1 在PLSQL Developer的sql window里输入select * from test for update; 2 按F8执行 3 打开锁, 再按一下加号. 鼠标点到第一列的列头,使全列成选中状态,然后粘贴,最后commit提交即可。(前提..._excel导入pl/sql

Git常用命令速查手册-程序员宅基地

文章浏览阅读83次。Git常用命令速查手册1、初始化仓库git init2、将文件添加到仓库git add 文件名 # 将工作区的某个文件添加到暂存区 git add -u # 添加所有被tracked文件中被修改或删除的文件信息到暂存区,不处理untracked的文件git add -A # 添加所有被tracked文件中被修改或删除的文件信息到暂存区,包括untracked的文件...

分享119个ASP.NET源码总有一个是你想要的_千博二手车源码v2023 build 1120-程序员宅基地

文章浏览阅读202次。分享119个ASP.NET源码总有一个是你想要的_千博二手车源码v2023 build 1120

【C++缺省函数】 空类默认产生的6个类成员函数_空类默认产生哪些类成员函数-程序员宅基地

文章浏览阅读1.8k次。版权声明:转载请注明出处 http://blog.csdn.net/irean_lau。目录(?)[+]1、缺省构造函数。2、缺省拷贝构造函数。3、 缺省析构函数。4、缺省赋值运算符。5、缺省取址运算符。6、 缺省取址运算符 const。[cpp] view plain copy_空类默认产生哪些类成员函数

推荐文章

热门文章

相关标签