基于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 ...
随机推荐
- CSS3新增文本属性实现图片点击切换效果
<!doctype html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
- Open-Falcon第四步安装Graph(小米开源互联网企业级监控系统)
安装Graph graph组件是存储绘图数据.历史数据的组件.transfer会把接收到的数据,转发给graph. cd /usr/local/open-falcon/graph mv cfg.exa ...
- 阿里消息队列中间件 RocketMQ 源码分析 —— Message 拉取与消费(上)
- Luogu P1001 A+B Problem
题目描述 输入两个整数a,b,输出它们的和(|a|,|b|<=10^9). 注意 1.pascal使用integer会爆掉哦! 2.有负数哦! 3.c/c++的main函数必须是int类型,而且 ...
- Linux - 简明Shell编程14 - 操作符(Operator)
脚本地址 https://github.com/anliven/L-Shell/tree/master/Shell-Basics 示例脚本及注释 #!/bin/bash echo "No c ...
- tcpdump抓包分析详解
說實在的,對於 tcpdump 這個軟體來說,你甚至可以說這個軟體其實就是個駭客軟體, 因為他不但可以分析封包的流向,連封包的內容也可以進行『監聽』, 如果你使用的傳輸資料是明碼的話,不得了,在 ro ...
- RabbitMQ --- Publish/Subscribe(发布/订阅)
目录 RabbitMQ --- Hello Mr.Tua RabbitMQ --- Work Queues(工作队列) 前言 在第二篇文章中介绍了 Work Queues(工作队列),它适用于把一个消 ...
- MySQL的left,substr,instr截取字符串函数使用实例
原表:select * from pagereferrer;如下: 如何在这张表中,查询出按 URL分类的信息 ,例如:211.95.60.43:8080算一类信息,并按百分比显示.预期结 ...
- SQL优化二(Sql性能调优)
一·.前言:这篇博文内容非原创,是我们公司的架构师给我们做技术培训的时候讲的内容,我稍微整理了下,借花献佛.这篇博文只是做一个大概的科普介绍,毕竟SQL优化的知识太大了,几乎可以用一本书来介绍.另外, ...
- Redis环境搭建(MacOS)
Redis是一个开源的key-value类型的存储系统,大部分数据存在于内存中,所有读写速度十分快.其支持的存储value数据类型有多种,如:strings.hashes.lists.sets.sor ...