基于NIOS-II的示波器:PART2 界面动态显示功能
本文所有的硬件基础以及工程参考来自魏坤示波仪,重新实现驱动并重构工程。
version 0.2 界面动态显示功能
界面显示功能原理
显示波形有如下两个方案:
- 每一帧直接重绘显示界面,再显示下一帧图形
- 用背景色覆盖上一帧图形,再显示被毁坏的坐标,最后显示下一帧图形
第一种解决方案代码比较简单,但是实际测试刷新缓慢(以肉眼可测的速度刷新),故选择第二种方案。
其中有几点需要注意的部分:
- 两个点直接需要连接起来,可以按照显示芯片写的顺序,依次填充y轴,来达到连线。
- 规划将波形数据保存在DISP数组中,再绘制的同时写入CLR数组中,供下一轮清理波形使用。
清波形函数如下:
/*
* 函数名:CLR_WAVE_CH1
* 功能:清除CH1波形
* 日期:2017-02-16
*/
void CLR_WAVE_CH1() {
unsigned int i, k;
for (i = 0; i < 400; i++) {
//确认CLR中有波形数据(在有效区域内)
if (CLR_DATA_CH1[i] > 60 && CLR_DATA_CH1[i] < 300) {
//清除这个点
set_addr(i + 11, CLR_DATA_CH1[i]);
send_data(WAVE_BACK_COLOR);
set_addr(i + 11, CLR_DATA_CH1[i] + 1);
send_data(WAVE_BACK_COLOR);
set_addr(i + 11, CLR_DATA_CH1[i] - 1);
send_data(WAVE_BACK_COLOR);
//下面清除连接两个点的线
if (i < 399) {
if (CLR_DATA_CH1[i] < CLR_DATA_CH1[i + 1]) {
set_addr(i + 11, CLR_DATA_CH1[i]);
IOWR_ALTERA_AVALON_PIO_DATA(LCD_RS_BASE, 1);
IOWR_ALTERA_AVALON_PIO_DATA(LCD_DATA_BASE, WAVE_BACK_COLOR);
for (k = CLR_DATA_CH1[i]; k < CLR_DATA_CH1[i + 1]; k++) {
IOWR_ALTERA_AVALON_PIO_DATA(LCD_WR_BASE, 0);
IOWR_ALTERA_AVALON_PIO_DATA(LCD_WR_BASE, 1);
}
} else {
set_addr(i + 11, CLR_DATA_CH1[i + 1]);
IOWR_ALTERA_AVALON_PIO_DATA(LCD_RS_BASE, 1);
IOWR_ALTERA_AVALON_PIO_DATA(LCD_DATA_BASE, WAVE_BACK_COLOR);
IOWR_ALTERA_AVALON_PIO_DATA(LCD_RS_BASE, 1);
IOWR_ALTERA_AVALON_PIO_DATA(LCD_DATA_BASE, WAVE_BACK_COLOR);
for (k = CLR_DATA_CH1[i + 1]; k < CLR_DATA_CH1[i]; k++) {
IOWR_ALTERA_AVALON_PIO_DATA(LCD_WR_BASE, 0);
IOWR_ALTERA_AVALON_PIO_DATA(LCD_WR_BASE, 1);
}
}
}
}
}
//显示坐标
display_area();
}
显示波形代码如下
void DISP_WAVE_CH1() {
unsigned int i, k;
//清除CH1波形
CLR_WAVE_CH1();
for (i = 0; i < 400; i++) {
//显示当前点
CLR_DATA_CH1[i] = DISP_DATA_CH1[i];
set_addr(i + 11, DISP_DATA_CH1[i]);
send_data(WAVE_COLOR_CH1);
// set_addr(i+11,DISP_DATA_CH1[i]+ 1);
set_y(DISP_DATA_CH1[i] + 1);
send_data(WAVE_COLOR_CH1);
set_addr(i + 11, DISP_DATA_CH1[i] - 1);
set_y(DISP_DATA_CH1[i] - 1);
// send_data(WAVE_COLOR_CH1);
//画连线
if (i < 399) {
if (DISP_DATA_CH1[i] < DISP_DATA_CH1[i + 1]) {
set_y(DISP_DATA_CH1[i]);
// set_addr(i+11,DISP_DATA_CH1[i]);
send_data(WAVE_COLOR_CH1);
for (k = DISP_DATA_CH1[i]; k < DISP_DATA_CH1[i + 1]; k++) {
IOWR_ALTERA_AVALON_PIO_DATA(LCD_WR_BASE, 0);
IOWR_ALTERA_AVALON_PIO_DATA(LCD_WR_BASE, 1);
// set_addr(i+11,k);
// send_data(WAVE_COLOR_CH1);
}
} else {
set_y(DISP_DATA_CH1[i + 1]);
send_data(WAVE_COLOR_CH1);
for (k = DISP_DATA_CH1[i + 1]; k < DISP_DATA_CH1[i]; k++) {
IOWR_ALTERA_AVALON_PIO_DATA(LCD_WR_BASE, 0);
IOWR_ALTERA_AVALON_PIO_DATA(LCD_WR_BASE, 1);
// set_addr(i+11,k);
// send_data(WAVE_COLOR_CH1);
}
}
}
}
}
类似的可以写出CH2的波形显示函数以及触发线等显示函数。
而其他的数据显示利用0.1版本的显示驱动即可轻松实现。
version 0.2.1 显示功能优化
在动态测试的过程中,发现这样通过调用打包中间层函数的方法,速度跟不上来,会出现频繁闪动的问题。
分析一个有以下几种可能
- 函数调用导致速度过慢
- 中间抽象层中有部分可以通过更少的时钟周期完成(利用像素的局部性)
将函数展开,以及部分不需要重新传输坐标的部分优化后解决了刷新的问题:
void CLR_WAVE_CH1() {
unsigned int i, k;
for (i = 0; i < 400; i++){
LCD_ILI9481_CMD(0x002b);
IOWR_ALTERA_AVALON_PIO_DATA(LCD_RS_BASE, 1);
IOWR_ALTERA_AVALON_PIO_DATA(LCD_WR_BASE, 0);
IOWR_ALTERA_AVALON_PIO_DATA(LCD_DATA_BASE, (i + 11) >> 8);
IOWR_ALTERA_AVALON_PIO_DATA(LCD_WR_BASE, 1);
IOWR_ALTERA_AVALON_PIO_DATA(LCD_WR_BASE, 0);
IOWR_ALTERA_AVALON_PIO_DATA(LCD_DATA_BASE, (i + 11) & 0x00ff);
//... 其余类似
}
}
Version 0.2版本演示
基于NIOS-II的示波器:PART2 界面动态显示功能的更多相关文章
- 【推荐图书】+ 基于Nios II的嵌入式SoPC系统设计与Verilog开发实例+C#入门经典等
[推荐图书]+ 基于Nios II的嵌入式SoPC系统设计与Verilog开发实例+C#入门经典等 3赞 发表于 2016/7/4 21:14:12 阅读(1921) 评论(3) 初次接触FPGA,到 ...
- 基于NIOS II的双端口CAN通信回环测试
基于NIOS II的双端口CAN通信回环测试 小梅哥编写,未经授权,严禁用于任何商业用途 说明:本稿件为初稿,如果大家在使用的过程中有什么疑问或者补充,或者需要本文中所述工程源文件,欢迎以邮件形式发送 ...
- FPGA回忆记事(一):基于Nios II的LED实验
实验一:基于Nios II的LED实验 一. 创建Quartus II工程 1.打开Quartus II环境.开始->程序->Altera->Quartus II 9.1. 2 ...
- [置顶]
基于FPGA的VGA简易显存设计&NIOS ii软核接入
项目简介 本项目基于Altera公司的Cyclone IV型芯片,利用NIOS II软核,2-port RAM与时序控制模块,实现64*48分辨率的显存(再大的显存板载资源m9k不够用) 实现效果如下 ...
- NIOS II 中直接调用Modelsim仿真
STEP1:创建一个工程,实现并编译该工程,编写TestBench文件. STEP2:设置启动Modelsim的路径 选择Nios II菜单Tools->Options..,在弹出的界面中,选择 ...
- NIOS ii 流水灯
为了做项目的前期验证工作,实验室购买了某开发板,下面是基于该板子的实现过程.作为笔记记录,供入门者参考. 1:创建一个Quartus II的工程 next选择器件,然后finish.我的器件是cycl ...
- 给NIOS II CPU增加看门狗定时器并使用
给NIOS II CPU增加看门狗定时器并使用 配置看门狗定时器: 设置计时溢出时间为1秒 计数器位宽为32位 勾选No Start/Stop control bits 勾选Fixed perio ...
- 关于Nios II的启动分析(转载)
原文地址:http://hi.baidu.com/goatdai/item/cc33671545d89243e75e06ad 常用到的存储器包括SDRMA.SRAM.FLASH.Onchip_memo ...
- 给NIOS II CPU添加一颗澎湃的心——sysclk的使用
给NIOS II CPU添加一颗澎湃的心——系统时钟的使用 本实验介绍如何在Qsys中添加一个定时器作为NIOS II的心跳定时器,并在NIOS II中软件编程使用该定时器. 将上一个实验watchd ...
随机推荐
- Final 关键字
1.涵义 最一般的意思就是声明 "这个东西不能改变".之所以要禁止改变,可能是考虑到两方面的因素:设计或效率. final 关键字可以用来修饰变量.方法和类,修饰变量表示变量不能被 ...
- Git时光机穿梭之删除文件
在Git中,删除也是一个修改操作,我们实战一下,先添加一个新文件test.txt到Git并且提交: $ git add test.txt $ git commit -m "add test. ...
- TASKCTL产品功能清单-转载
功能分类 功能描述 一级 二级 关系 调度控制 作业依赖关系调度 作业依赖关系调度是调度最基本的功能,指作业间具有顺序的运行,比如:a.b.c三个作业,只有当a完成后才运行b,b完成才能运行c 作业并 ...
- 脱壳练习之bitarts 5.0
运行界面 一开始不是PUSHAD,这里我们跟到PUSHAD指令处,按F7执行该指令,接着在寄存器窗口中定位到ESP寄存器的值,在其上面单击鼠标右键选择-Follow in Dump. 仅允许非商业转载 ...
- [bzoj4832]抵制克苏恩 概率dp
考试的时候打了个搜索,时间比较短,样例又非常的弱,实在不太清楚他这个到底是什么意思. 不过lc大神好腻害,讲解了一下非常的清楚了. f[i][j][k][l]表示第i次伤害(啊),一滴血j个,两滴血k ...
- Shell - 简明Shell编程
本文是对Shell脚本编程的总结和回顾,所有涉及的脚本均已做了基本的调试和验证. [toc] 测试环境信息 [root@CentOS7 ~]# uname -a Linux CentOS7 3.10. ...
- webdriver
导入模块: from selenium import webdriver from selenium.common.exceptions import NoSuchElementException 选 ...
- 基于 Laravel、Vue.js开发的全新社交系统----ThinkSNS+
什么是ThinkSNS+ ThinkSNS(简称TS)始于2008年,一款全平台综合性社交系统,为国内外大中小企业和创业者提供社会化软件研发及技术解决方案,目前最新版本为ThinkSNS+.新的产品名 ...
- init.ora, pfile, spfile
实例启动时,查找初始化参数文件的顺序为: spfile<sid>.oraspfile.orainit<sid>.ora 如果以上3个文件都不存在,则实例无法启动. init.o ...
- ActionBar+DrawerLayout实现网易新闻客户端首页
一.概述 随着android版本的不断的更新,google推出了越来越多的高级组件,采用这些官方组件我们可以方便的实现一些以前需要通过复杂编码或者使用第三方组件才能实现的效果,比如slidingmen ...