vivado验证ddr引脚_vivado下ddr3的读写和测试详解stormy
最近博主在根据例程做ddr3的读写测试,发现根本看不到好吧,虽然之前博主做过SDRAM的读写测试,但是ddr3更加复杂,时序写起来很吃⼒,所以需要⽤到vivado下⾃带的ip核。具体来看下⾯例化过程:
1.在ip核下搜索mig 双击打开
2.直接next 然后在当前界⾯修改你的ddr3ip核的名字
这⾥博主是因为已经例化了ip核,所以名字不能修改,然后next
3.这是要不要兼容芯⽚,不选,点击next
4.勾选你的存储器类型,我的是ddr3,点击next
5.
这个配置⽐较多,第⼀个时钟为ddr3实际⼯作的时钟,然后选择你的内存型号,数据宽度即可,点击next
easy going6.
然后输⼊时钟可能需要pll倍频得到,⼀般是200Mhz,这⾥注意看下最后⼀⾏的⽤户地址类型,它是由bank+row+column组成的,这个在后⾯的读写测试会进⼀步提到。
7.
时钟选择不差分,然后参考时钟为⽤户时钟。
8.下⾯就是默认next,然后就是分配管脚了,这个你买的开发板⼀般都会提⾼ucf⽂件,直接复制就⾏。
然后next,⽣成。
以上就是ip核的简单例化过程,这个步骤⽹上有很多类似的,博主就不⼀⼀讲解了,把精⼒放在读写测试这块。
⾸先来看⽼三样:ip核⽤户界⾯下的控制命令,读和写
这是控制命令,可以让⽤户来发送读或者写命令,需要注意的事只有当app_rdy和app_en同为⾼时才
有效,命令被发出。这⾥博主通过ila 上电分析发现app_rdy为ip核⾃⼰产⽣的输出信号,但是它并不是⼀直都是⾼电平,所以在后续的读写测试时需要判断,⾄于怎么判断,我们后⾯代加上电分析。
上⾯是写命令,可以看到当add_wdf_wren和add_wdf_end同为⾼时数据才能有效被写进去,同时app_wdf_rdy也要为⾼。需要注意的⼀点是,写数据和写命令此时不再有关系,为什么,因为写数据其实是通过fifo缓存,当写命令有效时,由于先进先出的特性会把它所对应数据给写⼊,当然这个很拗⼝,下⾯会给出⽰例
上⾯的是读过程,可以看出当读命令发出后需要⼀个延迟读数据才会有效。
下⾯来看代码进⾏讲解:
module mem_burst
#(
parameter MEM_DATA_BITS = 64,
parameter ADDR_BITS = 24
)
(
input rst, /*复位*/
英语四级分值分布情况input mem_clk, /*接⼝时钟*/
input rd_burst_req, /*读请求*/
input wr_burst_req, /*写请求*/
input[9:0] rd_burst_len, /*读数据长度*/
input[9:0] wr_burst_len, /*写数据长度*/
input[ADDR_BITS - 1:0] rd_burst_addr, /*读⾸地址*/
input[ADDR_BITS - 1:0] wr_burst_addr, /*写⾸地址*/
output rd_burst_data_valid, /*读出数据有效*/
output wr_burst_data_req, /*写数据信号*/
output[MEM_DATA_BITS - 1:0] rd_burst_data, /*读出的数据*/ input[MEM_DATA_BITS - 1:0] wr_burst_data, /*写⼊的数据*/ output rd_burst_finish, /*读完成*/
output wr_burst_finish, /*写完成*/
output burst_finish, /*读或写完成*/
///
output[ADDR_BITS-1:0] app_addr,
output[2:0] app_cmd,
output app_en,
output [MEM_DATA_BITS-1:0] app_wdf_data,
output app_wdf_end,
output [MEM_DATA_BITS/8-1:0] app_wdf_mask,
output app_wdf_wren,
input [MEM_DATA_BITS-1:0] app_rd_data,
input app_rd_data_end,
input app_rd_data_valid,
input app_rdy,
input app_wdf_rdy,
input ui_clk_sync_rst,
input init_calib_complete
);
assign app_wdf_mask = {MEM_DATA_BITS/8{1'b0}}; localparam IDLE = 3'd0;
localparam MEM_READ = 3'd1;
localparam MEM_READ_WAIT = 3'd2;
localparam MEM_WRITE = 3'd3;
localparam MEM_WRITE_WAIT = 3'd4;
localparam READ_END = 3'd5;
localparam WRITE_END = 3'd6;
localparam MEM_WRITE_FIRST_READ = 3'd7;
/*parameter IDLE = 3'd0;
parameter MEM_READ = 3'd1;
parameter MEM_READ_WAIT = 3'd2;
parameter MEM_WRITE = 3'd3;
parameter MEM_WRITE_WAIT = 3'd4;
parameter READ_END = 3'd5;
parameter WRITE_END = 3'd6;
parameter MEM_WRITE_FIRST_READ = 3'd7;*/
reg[2:0] state;
reg[9:0] rd_addr_cnt;
reg[9:0] rd_data_cnt;
reg[9:0] wr_addr_cnt;
人际沟通能力reg[9:0] wr_data_cnt;
reg[2:0] app_cmd_r;
reg[ADDR_BITS-1:0] app_addr_r;
a reasonreg app_en_r;
reg app_wdf_end_r;
典雅什么意思
reg app_wdf_wren_r;
assign app_cmd = app_cmd_r;
assign app_addr = app_addr_r;
assign app_en = app_en_r;
assign app_wdf_end = app_wdf_end_r;
assign app_wdf_data = wr_burst_data;
assign app_wdf_wren = app_wdf_wren_r & app_wdf_rdy; assign rd_burst_finish = (state == READ_END);
assign wr_burst_finish = (state == WRITE_END);
assign burst_finish = rd_burst_finish | wr_burst_finish;
assign rd_burst_data = app_rd_data;
assign rd_burst_data_valid = app_rd_data_valid;
assign wr_burst_data_req = (state == MEM_WRITE) & app_wdf_rdy ; always@(podge mem_clk or podge rst)
begin
if(rst)
begin
app_wdf_wren_r <= 1'b0;
end
el if(app_wdf_rdy)
app_wdf_wren_r <= wr_burst_data_req;
end
always@(podge mem_clk or podge rst)
begin
if(rst)
begin
state <= IDLE;
app_cmd_r <= 3'b000;
app_addr_r <= 0;
app_en_r <= 1'b0;
rd_addr_cnt <= 0;
rd_data_cnt <= 0;
wr_addr_cnt <= 0;
wr_data_cnt <= 0;
app_wdf_end_r <= 1'b0;
end
oral是什么意思el if(init_calib_complete === 1'b1)
begin
ca(state)
IDLE:
begin
if(rd_burst_req)
begin
state <= MEM_READ;
app_cmd_r <= 3'b001;
app_addr_r <= {rd_burst_addr,3'd0}; app_en_r <= 1'b1;
end
冰雪奇缘2有没有彩蛋el if(wr_burst_req)
begin
state <= MEM_WRITE;
app_cmd_r <= 3'b000;
app_addr_r <= {wr_burst_addr,3'd0}; app_en_r <= 1'b1;
wr_addr_cnt <= 0;
app_wdf_end_r <= 1'b1;
wr_data_cnt <= 0;
end
end
MEM_READ:
begin
tangguif(app_rdy)
begin
app_addr_r <= app_addr_r + 8;
if(rd_addr_cnt == rd_burst_len - 1) begin
state <= MEM_READ_WAIT;
rd_addr_cnt <= 0;
app_en_r <= 1'b0;
高一英语辅导书end
el
rd_addr_cnt <= rd_addr_cnt + 1;
end
if(app_rd_data_valid)
begin
//app_addr_r <= app_addr_r + 8;
if(rd_data_cnt == rd_burst_len - 1) begin