FPGA实现“打字机”(VGA & UART)
看到标题中的“打字机”三个字,你是不是脑补了下面这幅图像。这是二战电影中常出现的道具,现在恐怕都见不到了。

●电影道具“打字机”
我要实现的当然不是这个样子,只是功能上与之相似。先让你们看看实现的效果,直接上图。

●这是串口发的字符串

●显示屏显示的字符
之所以要写这个“打字机”工程,那是因为我在学习FPGA的道路上,它是我重要的一关。我一开始学FPGA,是从数字电路开始入门的,然后就是学习使用QuartusII,编写Verilog代码。我写的第一个工程是数字时钟(6个数码管两个一组实现时钟、分钟、秒钟的计时),通过它我学会了基本的逻辑(时序逻辑、组合逻辑,以及我们用的最频繁的译码器和选择器)。接下来就是这个“打字机”了,其实一开始这是学长给我布置的一个任务(现在想想学长也是用心良苦),通过它我基本上领会了如何驱动外设以及用各个外设组合成我们需要的工程(也就是模块化设计)。我相信,对于很多还在朝FPGA入门的同学来说,通过这个工程,你们的能力能获得显著提升。
好了,话就不多说了,接下来就来看看这个工程是怎么实现的。我们先来看看RTL视图:

●RTL视图
从该图中可以看到,这里一共有5个模块:PLL模块、UARTRX接收模块、DISMEM显存模块、GETZIMO字模译码模块、VGA显示模块。PLL模块产生VGA(800*600@60Hz)所需的40M时钟,UARTRX模块接收上位机发来的数据,显存模块保存需要显示的字符,字模模块将显存的字符译码成VGA模块需要的数据。PLL、UARTRX、VGA这三个模块就不讲了,这是学习FPGA必然要遇到的模块(网上有很多代码),接下来就给大家重点分析一下显存模块和字模译码模块。
显存模块,顾名思义就是仿造显卡的显存,将显示的内容保存起来,当显示模块需要的时候再取出来。当我们按顺序将UART接收到的字符放入RAM中时,有一个问题,如果写满了RAM怎么办?当然,可以这么做,从头开始写,一直循环。那么读操作也按顺序读就会有问题了,这时如果写满了,之前显示的内容就突然没了。可是我们希望当写满时所有行都能往上卷一行,这样就和我们平时打字的效果一样了。解决这个问题的办法是改变读地址和写地址的映射关系。通常,读写地址是这么映射的:

为了解决该问题,映射关系改成这样:

每当写满一页,读地址就往上卷动一行(也就是减去一行)。具体的代码就是:
//从串口来一个数据就加1,直到计满清零
always @ (posedge CLK_50M or negedge RST_N)
begin
if(!RST_N)
RAM_WP <= 'h0;
else
RAM_WP <= RAM_WP_N;
end always @ (*)
begin
if(RAM_WP == 'd3699 && DATAFLAG)
RAM_WP_N = 'h0;
else if(DATAFLAG)
RAM_WP_N = RAM_WP + 'h1;
else
RAM_WP_N = RAM_WP;
end //RAM_WP对100取模得RAM_WP100,可以作为写指针的列
always @ (posedge CLK_50M or negedge RST_N)
begin
if(!RST_N)
RAM_WP100 <= 'h0;
else
RAM_WP100 <= RAM_WP100_N;
end
always @ (*)
begin
if(RAM_WP100 == 'd99 && DATAFLAG)
RAM_WP100_N = 'h0;
else if(DATAFLAG)
RAM_WP100_N = RAM_WP100 + 'h1;
else
RAM_WP100_N = RAM_WP100;
end //以下实现卷行 //判断页尾,判断成功则一直保持
always @ (posedge CLK_50M or negedge RST_N)
begin
if(!RST_N)
sroll_flag <= 'b0;
else
sroll_flag <= sroll_flag_n;
end
always @ (*)
begin
if(RAM_WP == 'd3699 && DATAFLAG)
sroll_flag_n = 'b1;
else
sroll_flag_n = sroll_flag;
end
字模译码模块就是将从显存RAM中的字符译码得到VGA需要显示的数据,这是因为字符是不能直接给VGA显示的,所以有这个译码的过程。译码过程需要根据显示的位置得到VGA显示的像素,也就是根据字符从ROM中(里面存有字模信息)取得字符的字模,其过程见代码:
//得到需要输出的字模数据的哪行,打一拍
always @ (posedge CLK_50M or negedge RST_N)
begin
if(!RST_N)
linechar <= 'h0;
else
linechar <= position_VS[:];
end
//得到该行数据
GETLINE getlineA
(
.CLK_50M (CLK_50M ),
.CHAR (CHAR ),
.linechar (linechar ),
.dataline (dataline )
); //得到需要输出字模数据的某列
always @ (posedge CLK_50M or negedge RST_N)
begin
if(!RST_N)
bitcnt <= 'h0;
else
bitcnt <= bitcnt_n;
end always @ (posedge CLK_50M or negedge RST_N)
begin
if(!RST_N)
bitcnt_n <= 'h0;
else
bitcnt_n <= position_HS[:];
end //得到字模数据某行某列上的值,并将这一位值译成8位数据(8'hff或8'h0)
DECODEPIXEL decodeA
(
.datain (dataline ),
.bitcnt (bitcnt ),
.dataout (databit )
); assign DATAOUT = databit;
这个“打字机”的工作流程可以通过下图来理解,数据由上位机发过来,UART模块接收,当接收到一个完整字符时,将这个字符写入到RAM。RAM中的数据读出后经过ROM译码,最后通过VGA送给显示器显示。

●字符信息流
“打字机”的具体实现就是上面将的这些,有兴趣的朋友可以在此基础上改进,改进的内容可以是增加控制字符的操作,也可以是将显示的字符保存到SDRAM中这样就可以容纳更多的字符了。另外,需要源码的朋友可以在这个地址下载: https://pan.baidu.com/s/1jIqnI1C。
让大家看一下我的实物,最近入手了一块锆石科技出品的A4开发板。在这块开发板上搭载了一颗Altera公司的EP4CE10F17C8,功能强大并且外设齐全,非常适合初学者。

需要说明一下,这篇文章是仅仅我发的第一篇博文,而且只是一个小工程,也是让大家在学习FPGA的过程中有一个很好的练手的机会。在后面,我还会定期推出更多的自己在FPGA学习道路上的经验心得,而且绝对是干货(比如USB2.0、USB3.0、DDR2、SDcard、摄像头等等)哦!小伙伴们可以持续关注我,更多惊喜等着你。
FPGA实现“打字机”(VGA & UART)的更多相关文章
- 模拟摄像头解码模块最新测试 TVP5150模块 FPGA+SDRAM+TVP5150+VGA 实现PAL AV输入 VGA视频输出
模拟摄像头解码模块最新测试 TVP5150模块 FPGA+SDRAM+TVP5150+VGA 实现PAL AV输入 VGA视频输出 测试使用电视机顶盒的AV模拟信号输入,VGA显示器输出测试,效 ...
- 从FPGA搞定OV7670 VGA显示 移植到 STM32F10x TFT显示 总结及疑问(高手请进)
OV7670不愧是最便宜的摄像头了最大显示像素:640*480(在VGA显示器上显示效果还不赖,用usb模块采集显示依然显著) 第一步:VGA显示 视频图像(实时)FPGA+SDRAM+OV7670= ...
- 基于FPGA的VGA显示实验设计
基于FPGA的VGA显示实验设计 成果展示(优酷视频): 视频: 基于FPGA的VGA显示技术(手机控制) http://v.youku.com/v_show/id_XNjk4ODE3ODUy.htm ...
- 202-基于TI DSP TMS320C6678、Xilinx K7 FPGA XC7K325T的高速数据处理核心板
该DSP+FPGA高速信号采集处理板由我公司自主研发,包含一片TI DSP TMS320C6678和一片Xilinx FPGA K7 XC72K325T-1ffg900.包含1个千兆网口,1个FMC ...
- VGA调试心得
以前自己调试过视频信号,无非就时钟加行场同步加数据线,如果视频信号出问题,第一看现象,第二测频率,反正出问题不是消隐信号出问题,就是时钟频率出问题.通过这种方式也调试成功过几个显示屏,然后就以为自己对 ...
- 基于TI DSP TMS320C6678、Xilinx K7 FPGA XC7K325T的高速数据处理核心板
一.板卡概述 该DSP+FPGA高速信号采集处理板由我公司自主研发,包含一片TI DSP TMS320C6678和一片Xilinx FPGA K7 XC72K325T-1ffg900.包含1个千兆网口 ...
- 数控AGC实现(转)
相关链接: 一种混合式高动态范围AGC算法与FPGA实现 http://www.sohu.com/a/221438387_781333 基于FPGA的快速自动增益控制系统设计 ...
- MATLAB串口操作和GUI编程
程序说明 V1.0 2015/2/08 MATLAB串口操作和GUI编程 概述 本文介绍了程序AD9512_Serial_GUI的编程思路和功能.该程序设计到MATLAB的图像用户界面编程的基 ...
- 【扫盲贴】为什么屏幕分辨率是 640x480
本文原地址:http://www.easyx.cn/skills/View.aspx?id=172 常见的屏幕分辨率很奇怪,为什么总用一些不零不整的数字?比如以前最常见的分辨率是 640x480,当初 ...
随机推荐
- windows 配置 Scheme + Emacs 编程环境
软件下载列表: Emacs Racket (这里使用 Racket ,更加方便,便于后面配置 Emacs) 配置 安装好 Emacs 后,在 C:\Users\用户名\AppData\Roaming\ ...
- keyup实现在输入状态不发送搜索请求,停止输入后发送
个人需求:通过keyup事件配合后台elasticsearch(弹性搜索),用户在输入状态不发送请求,等停止输入后发送请求. 这是个思考笔记,因为项目临时需要弹性搜索功能,所以临时想了这么个法子,方法 ...
- 面向对象(java菜鸟的课堂笔记)
类:相同的东西放在一起 分为属性和动作: 把一组或多组事物相同的特性的描述==>类 属性和动作被称为成员: //声明类的属性信息 public class **{ String name: ...
- Python之正则表达式(re模块)
本节内容 re模块介绍 使用re模块的步骤 re模块简单应用示例 关于匹配对象的说明 说说正则表达式字符串前的r前缀 re模块综合应用实例 正则表达式(Regluar Expressions)又称规则 ...
- 【2017-06-06】Ajax完整结构、三级联动的制作
一.Ajax完整结构 $.ajax({ url:"Main.ashx", data:{}, dataType:"json", type:"post&q ...
- springmvc 添加@ResponseBody
1.添加ResponseBody之后的话 返回字符串的时候 就是一个字符串. @RequestMapping(value = "/{bookId}/detail.do",metho ...
- if中可以使用那些作为判断条件呢?
在所有编程语言中if是最长用的判断之一,但在js中到底哪些东西可以在if中式作为判断表达式呢? 例如如何几行,只是少了一个括号,真假就完全不同,到底表示什么含义呢 ? 1 2 3 4 5 6 7 8 ...
- Tenacity——Exception Retry 从此无比简单
Python 装饰器装饰类中的方法这篇文章,使用了装饰器来捕获代码异常.这种方式可以让代码变得更加简洁和Pythonic. 在写代码的过程中,处理异常并重试是一个非常常见的需求.但是如何把捕获异常并重 ...
- MAC下解决eclipse卡顿或者运行慢的问题
提示:假设你已经装了固态硬盘,并且有至少8Gb的内存.如果没有的话,带来的性能提升可能不大. 1.eclipse中加载的SDK数量过多会导致程序运行缓慢,解决方法删除plaforms下面用不到的SDK ...
- 一个编程菜鸟的进阶之路(C/C++)
学编程是一条不归路,但我义无反顾.只能往前冲,知道这个过程是痛苦的,所以我开通这个博客,记录自己在编程中遇到的问题和心得,一是希望可以帮助跟我一样遇到同样问题的人,二是把这作为对自己的勉励及回忆: