用嵌入式块RAM IP核配置一个双口RAM
本次设计源码地址:http://download.csdn.net/detail/noticeable/9914173
实验现象:通过串口将数据发送到FPGA 中,通过quartus II 提供的in system memory content editor 工具查看RAM中接收到的数据,当需要是,按下按键KEY0,将FPGA 的RAM 中存储的数据通过串口发送出去。
知识点:
(1)存储器IP核的使用(2)in system memory content editor 内存查看工具的使用
系统结构框图如下,如何绘制在http://www.cnblogs.com/noticeable/p/7248404.html可以进行查看
设计过程:复制之前写的uart工程,打开,添加menmoryIP核
然后直接next,直到finish,此时就可以将IP核添加完毕了。
直接将IP核生成的文件设置为顶层文件,编写testbench文件对其进行仿真
`timescale 1ns/1ns
`define clock_period module dua_ram_tb;
reg clock;
reg [:]data;
reg [:]rdaddress;
reg [:]wraddress;
reg wren;
wire [:]q;
integer i;
dua_ram dua_ram0(
.clock(clock),
.data(data),
.rdaddress(rdaddress),
.wraddress(wraddress),
.wren(wren),
.q(q)
);//端口连接 initial clock=;
always#(`clock_period/)clock=~clock; initial begin
data=;
rdaddress=;
wraddress=;
wren=;
#(`clock_period*+);
for(i=;i<=;i=i+)begin // 写操作
wren=;
data=-i;
wraddress=i;
#(`clock_period);
end
wren=;
#(`clock_period*+);//延迟一段时间后进行读操作
for(i=;i<=;i=i+)begin // 读操作
rdaddress=i;
#(`clock_period);
end
#;
$stop;
end
endmodule
设置仿真脚本,点击仿真,进行前仿
仿真完了,可以对双口RAM有一个基本了解了,下面就将其添加到之前写的uart工程中
新建一个uart_dpram.v文件,并设置其为顶层文件,代码按照结构图进行编辑如下:
module uart_dpram(
clk,
rst_n,
key_in,
rs232_rx,
rs232_tx);
input clk;
input rst_n;
input key_in;
input rs232_rx; //串口读取端口
output rs232_tx; //串口写出端口 wire key_flag; //按键检测标志
wire key_state; //按键状态标志
wire rx_done; //读取完成标志
wire tx_done; //写出完成标志
wire send_en; //写出使能
wire [:]rdaddress,wraddress; //读地址和写地址
wire wren; //写入使能
wire [:]rx_data,tx_data; //待读取数据与发送数据 uart_tx uart_tx_a(
.clk(clk),
.rst_n(rst_n),
.send_en(send_en),
.baud_set('d0),
.tx_done(tx_done),
.rs232_tx(rs232_tx),
.data_byte(tx_data),
.uart_state()
); uart_rx uart_rx_a(
.clk(clk),
.rs232_rx(rs232_rx),
.baud_set('d0),
.rst_n(rst_n),
.data_byte(rx_data),
.rx_done(rx_done)
);
key_filter key_filter_a(
.clk(clk),
.rst_n(rst_n),
.key_in(key_in),
.key_flag(key_flag),
.key_state(key_state)
); dua_ram dua_ram_a(
.clock(clk),
.data(rx_data), //存储器入口是读取的数据
.rdaddress(rdaddress),
.wraddress(wraddress),
.wren(wren),
.q(tx_data) //存储器的出口是要写出的数据
);//端口连接
ctrl ctrl_a(
.clk(clk),
.rst_n(rst_n),
.key_flag(key_flag),
.key_state(key_state),
.rx_done(rx_done),
.tx_done(tx_done),
.rdaddress(rdaddress),
.wraddress(wraddress),
.wren(wren),
.send_en(send_en)
); endmodule
如代码中所说,ctrl模块没有定义,编译肯定是会报错的,下面继续编写ctrl 模块,新建一个ctrl.v文件,编辑如下:
module ctrl(
clk,
rst_n,
key_flag,
key_state,
rx_done,
tx_done,
rdaddress,
wraddress,
wren,
send_en
);//控制模块 input clk;
input rst_n;
input key_flag;
input key_state;
input rx_done;
input tx_done;
output reg[:]rdaddress;
output reg[:]wraddress;
output wren; //写入使能信号
output reg send_en; //发送使能 assign wren=rx_done; //读取完成的信号即可以写入了
reg do_send; //发送标志
reg r0_send_done,r1_send_done; //发送缓冲寄存器 always@(posedge clk or negedge rst_n)//写地址
if(!rst_n)
wraddress<='d0;
else if(rx_done) //每读取完一次,写入地址+1
wraddress<=wraddress+'d1;
else
wraddress<=wraddress; always@(posedge clk or negedge rst_n)
if(!rst_n)
do_send<=;
else if(key_flag&&!key_state) //如果按键按下一次,发送状态改变一次
do_send<=~do_send; always@(posedge clk or negedge rst_n)//读地址
if(!rst_n)
rdaddress<='d0;
else if(do_send&&tx_done) //发送状态为发送,且1byte数据已经发送完成了,则读地址+1
rdaddress<=rdaddress+'d1; always@(posedge clk or negedge rst_n)///ram读有两拍的延迟,所以加两级寄存器对发送使能进行缓存
if(!rst_n) begin
r0_send_done<='b0;
r1_send_done<='b0;
end
else begin
r0_send_done<=(do_send&&tx_done);
r1_send_done<=r0_send_done;
end always@(posedge clk or negedge rst_n)//发送使能
if(!rst_n)
send_en<=;
else if(key_flag&&!key_state)
send_en<='b1;
else if(r1_send_done)
send_en<=;
else
send_en<=; endmodule
编写testbench文件进行仿真,将test9中的keymodule 复制到本工程的testbench文件夹下,并添加到工程中
`timescale 1ns/1ns
`define clock_period
module uart_dpram_tb;
reg clk;
reg rst_n;
wire key_in;
wire rs232_rx;
wire rs232_tx;
wire tx_done;
reg [:]data_byte_t;
reg send_en;
wire [:]baud_set;
reg press; assign baud_set = 'd0;
uart_dpram uart_dpram_1(
.clk(clk),
.rst_n(rst_n),
.key_in(rst_n),
.rs232_rx(rs232_rx),
.rs232_tx(rs232_tx)); uart_tx uart_tx1(
.clk(clk),
.rst_n(rst_n),
.send_en(send_en),
.baud_set(baud_set),
.tx_done(tx_done),
.rs232_tx(rs232_rx),
.data_byte(data_byte_t),
.uart_state()
); key_module key_module1(
.press(press),
.key(key_in)
);
initial clk=;
always#(`clock_period/) clk=~clk;
//发送数据
initial begin
rst_n = 'b0;
press = ;
data_byte_t = 'd0;
send_en = 'd0;
#(`clock_period*+)
rst_n='b1;
#(`clock_period*+) data_byte_t<='haa;
send_en<='d1;
#(`clock_period)
send_en<='d0; @(posedge tx_done)//等待传输完成的上升沿 #(`clock_period*)//重新发送
data_byte_t<='h55;
send_en<='d1;
#(`clock_period)
send_en<='d0;
@(posedge tx_done)//等待传输完成的上升沿
#(`clock_period*)//重新发送
data_byte_t<='hcc;
send_en<='d1;
#(`clock_period)
send_en<='d0; @(posedge tx_done)//等待传输完成的上升沿
#(`clock_period*)//重新发送
data_byte_t<='hff;
send_en<='d1;
#(`clock_period)
send_en<='d0; //按下按键读取发送出去的内容
@(posedge tx_done) #(`clock_period*); press = ;
#(`clock_period*)
press = ;
#(`clock_period*); end
endmodule
添加路径如下:
点击仿真,仿真结果如下
引脚配置,烧写程序,进行扳级验证,这里GPIO0_D0连接到USB_TTL的TXD,GPIO0_D1连接到USB_TTL 的RXD即可。
打开串口调试软件,连接USB-TTL模块,发送数据,按下按键KEY_1,FPGA不断把刚才发到ram中的数据读取出来,再按一下,停止发送,效果图如下所示
用嵌入式块RAM IP核配置一个双口RAM的更多相关文章
- 第7讲 SPI和RAM IP核
学习目的: (1) 熟悉SPI接口和它的读写时序: (2) 复习Verilog仿真语句中的$readmemb命令和$display命令: (3) 掌握SPI接口写时序操作的硬件语言描述流程(本例仅以写 ...
- SPI和RAM IP核
学习目的: (1) 熟悉SPI接口和它的读写时序: (2) 复习Verilog仿真语句中的$readmemb命令和$display命令: (3) 掌握SPI接口写时序操作的硬件语言描述流程(本例仅以写 ...
- 7 Series GTP IP核使用总结 IP核配置篇
FPGA内嵌收发器相当于以太网中的PHY芯片,但更灵活更高效,线速率也在随着FPGA芯片的发展升级.本文对7系列FPGA内部高速收发器GTP IP核的配置和使用做些简单的总结,以备后续回顾重用.本文是 ...
- 单口 RAM、伪双口 RAM、真双口 RAM、单口 ROM、双口 ROM 到底有什么区别呢?
打开 IP Catalog,搜索 Block Memory Generator,即可看到其 Memory Type 可分为 5 中,分别是单口 RAM(Single Port RAM).伪双口 RAM ...
- 双口RAM,值得研究
在FPGA设计过程中,使用好双口RAM,也是提高效率的一种方法. 官方将双口RAM分为简单双口RAM和真双口RAM. 简单双口RAM只有一个写端口,一个读端口. 真双口RAM分别有两个写端口和两个读端 ...
- FMC与FPGA双口ram通讯
硬件环境:ARM+FPGA通过FMC互联,STM32F767和 EP4CE15F23I7 FMC设置,STM的系统时钟HCLK为216MHz /* FMC initialization functio ...
- 单口RAM、双口RAM、FIFO
单口与双口的区别在于,单口只有一组数据线与地址线,因此读写不能同时进行.而双口有两组数据线与地址线,读写可同时进行.FIFO读写可同时进行,可以看作是双口. 双口RAM分伪双口RAM(Xilin ...
- 【iCore1S 双核心板_FPGA】例程十七:基于双口RAM的ARM+FPGA数据存取实验
实验现象: 核心代码: module DUAL_PORT_RAM( input CLK_12M, inout WR, input RD, input CS0, :]A, :]DB, output FP ...
- xilinx IP核配置,一步一步验证Xilinx Serdes GTX最高8.0Gbps
版权声明:本文为博主原创文章,未经博主允许不得转载. https://blog.csdn.net/u010161493/article/details/77658599 目录(?)[+] 之前 ...
随机推荐
- expect login ssh
#!/usr/bin/expect -f set pwffd [lindex $argv 0] spawn ssh cmesvr2i expect "*password:" sen ...
- docker命令相关
进入容器 容器已经启动 docker exec -it ece7b58a2a04 /bin/sh 容器未启动 docker run -it zzzzz/edas:v1 sh 检查容器 docker i ...
- 学习node.js 第2篇 介绍node.js 安装
Node.js - 环境安装配置 如果愿意安装设置Node.js环境,需要计算机上提供以下两个软件: 一.文本编辑器 二.Node.js二进制安装包 文本编辑器 这将用来编写程序代码. 一些编辑器包括 ...
- 初试Python语法小试牛刀之冒泡排序
Python很火,心里很慌,没吃过猪肉,也要见见猪走路. 看了几天Python的语法,大概初步了解了一点点,https://www.liaoxuefeng.com/wiki/0014316089557 ...
- 云笔记项目-Java反射知识学习
在云笔记项目中,补充了部分反射的知识,反射这一部分基础知识非常重要,前面学习的框架Spring和MyBatis读取xml配置文件创建对象,以及JDBC加载驱动等都用了反射,但只知道有这个东西,具体不知 ...
- extJs相关名字解释
1.initComponent 初始化部件启动 2.defaults : Object defaults属性可以包含任意个name/value属性对,这些属性将会被添加到每一个元素中...例如, ...
- git配置正确且权限已开但是pull或push提示无权限
因为之前提示输入用户名和密码时输入错误,之后就一直权限认证失败.这种情况下在git bash中输入: git config --system --unset credential.helper 就会重 ...
- CSS3 white-space属性
white-space 属性设置如何处理元素内的空白. 可能的值 值 描述 normal 默认.空白会被浏览器忽略. pre 空白会被浏览器保留.其行为方式类似 HTML 中的 <pre> ...
- f5 主备模式切换
f5 主备模式 主机down自动切换到备 原主机重新启动,自动切换到原主机
- [leetcode]37. Sudoku Solver 解数独
Write a program to solve a Sudoku puzzle by filling the empty cells. A sudoku solution must satisfy ...