当前位置:   article > 正文

使用VIVADO中的MIG控制DDR3(AXI接口)二——用AXI4读写BRAM测试_axi ddr mig

axi ddr mig

        上篇文章主要讲了一些关于AXI的知识,有了这些理论,让我们进行一些简单的实验测试,加深对AXI协议的理解。本次实验使用的平台是VIVADO2019.1。

       1. 首先创建一个工程,名称和路径自己决定。

        2.然后按照图中所示,点击新建和打包一个新IP核

3.点击NEXT

4.选择一个带AXI4接口的IP核,然后点击NEXT

 5.记得修改名称,不然后边不好修改,路径可以直接放在工程根目录下

 6.名称我们进行修改,然后选用FULL接口的AXI,选择IP核为主机,数据位宽选32bits

 7.我们选择编辑此IP,我们便成功调用IP

 8.IP设置完成后,我们可以看到里边的.v代码,主要就是一个测试模块,它实现的功能就是利用AXI4接口向外发送1024个数据,然后再将数据读回,通过对比来验证AXI4这个IP核的功能。我对生成的代码做了一些注释,大家可以对照理解,代码不需要动,可以直接使用。

  1. `timescale 1 ns / 1 ps
  2. module AXI4_v1_0_AXI_MASTER #
  3. (
  4. // Users to add parameters here
  5. // User parameters ends
  6. // Do not modify the parameters beyond this line
  7. // Base address of targeted slave 基地址
  8. parameter C_M_TARGET_SLAVE_BASE_ADDR = 32'h40000000,
  9. // Burst Length. Supports 1, 2, 4, 8, 16, 32, 64, 128, 256 burst lengths 突发长度
  10. parameter integer C_M_AXI_BURST_LEN = 16,
  11. // Thread ID Width 写ID宽度
  12. parameter integer C_M_AXI_ID_WIDTH = 1,
  13. // Width of Address Bus 写地址宽度
  14. parameter integer C_M_AXI_ADDR_WIDTH = 32,
  15. // Width of Data Bus 写数据宽度
  16. parameter integer C_M_AXI_DATA_WIDTH = 32,
  17. // Width of User Write Address Bus
  18. parameter integer C_M_AXI_AWUSER_WIDTH = 0,
  19. // Width of User Read Address Bus
  20. parameter integer C_M_AXI_ARUSER_WIDTH = 0,
  21. // Width of User Write Data Bus
  22. parameter integer C_M_AXI_WUSER_WIDTH = 0,
  23. // Width of User Read Data Bus
  24. parameter integer C_M_AXI_RUSER_WIDTH = 0,
  25. // Width of User Response Bus
  26. parameter integer C_M_AXI_BUSER_WIDTH = 0
  27. )
  28. (
  29. // Users to add ports here
  30. // User ports ends
  31. // Do not modify the ports beyond this line
  32. //用户IO
  33. // Initiate AXI transactions 写开始信号
  34. input wire INIT_AXI_TXN,
  35. // Asserts when transaction is complete写完成信号
  36. output wire TXN_DONE,
  37. // Asserts when ERROR is detected 数据检测信号
  38. output reg ERROR,
  39. // Global Clock Signal. 全剧时钟
  40. input wire M_AXI_ACLK,
  41. // Global Reset Singal. This Signal is Active Low 全局复位信号
  42. input wire M_AXI_ARESETN,
  43. // Master Interface Write Address ID 写地址ID
  44. output wire [C_M_AXI_ID_WIDTH-1 : 0] M_AXI_AWID,
  45. // Master Interface Write Address 写地址
  46. output wire [C_M_AXI_ADDR_WIDTH-1 : 0] M_AXI_AWADDR,
  47. // Burst length. The burst length gives the exact number of transfers in a burst
  48. output wire [7 : 0] M_AXI_AWLEN, //写长度
  49. // Burst size. This signal indicates the size of each transfer in the burst
  50. output wire [2 : 0] M_AXI_AWSIZE, //写宽度
  51. // Burst type. The burst type and the size information,
  52. // determine how the address for each transfer within the burst is calculated.
  53. output wire [1 : 0] M_AXI_AWBURST, //写类型
  54. // Lock type. Provides additional information about the
  55. // atomic characteristics of the transfer.
  56. output wire M_AXI_AWLOCK, //独占传输类型
  57. // Memory type. This signal indicates how transactions
  58. // are required to progress through a system.
  59. output wire [3 : 0] M_AXI_AWCACHE, //存储类型,指明是否缓存
  60. // Protection type. This signal indicates the privilege
  61. // and security level of the transaction, and whether
  62. // the transaction is a data access or an instruction access.
  63. output wire [2 : 0] M_AXI_AWPROT, //安全保护
  64. // Quality of Service, QoS identifier sent for each write transaction.
  65. output wire [3 : 0] M_AXI_AWQOS, //服务质量QOS
  66. // Optional User-defined signal in the write address channel. 用户自定义
  67. output wire [C_M_AXI_AWUSER_WIDTH-1 : 0] M_AXI_AWUSER,
  68. // Write address valid. This signal indicates that
  69. // the channel is signaling valid write address and control information.
  70. output wire M_AXI_AWVALID, // 写地址发送有效,主机发给从机
  71. // Write address ready. This signal indicates that
  72. // the slave is ready to accept an address and associated control signals
  73. input wire M_AXI_AWREADY, //写地址接收有效,从机发给主机
  74. // Master Interface Write Data. 写数据 32位
  75. output wire [C_M_AXI_DATA_WIDTH-1 : 0] M_AXI_WDATA,
  76. // Write strobes. This signal indicates which byte
  77. // lanes hold valid data. There is one write strobe
  78. // bit for each eight bits of the write data bus. 频闪信号32/8 = 4位
  79. output wire [C_M_AXI_DATA_WIDTH/8-1 : 0] M_AXI_WSTRB,
  80. // Write last. This signal indicates the last transfer in a write burst.
  81. output wire M_AXI_WLAST, //写数据最后一位
  82. // Optional User-defined signal in the write data channel.
  83. output wire [C_M_AXI_WUSER_WIDTH-1 : 0] M_AXI_WUSER, //用户自定义
  84. // Write valid. This signal indicates that valid write
  85. // data and strobes are available 写数据发送有效,主到从
  86. output wire M_AXI_WVALID,
  87. // Write ready. This signal indicates that the slave
  88. // can accept the write data. //写数据接收有效,从到主
  89. input wire M_AXI_WREADY,
  90. // Master Interface Write Response. 写响应ID
  91. input wire [C_M_AXI_ID_WIDTH-1 : 0] M_AXI_BID,
  92. // Write response. This signal indicates the status of the write transaction.
  93. input wire [1 : 0] M_AXI_BRESP, //写响应信号
  94. // Optional User-defined signal in the write response channel
  95. input wire [C_M_AXI_BUSER_WIDTH-1 : 0] M_AXI_BUSER, // 用户自定义
  96. // Write response valid. This signal indicates that the
  97. // channel is signaling a valid write response.
  98. input wire M_AXI_BVALID, //写响应信号发送有效,从到主
  99. // Response ready. This signal indicates that the master
  100. // can accept a write response.
  101. output wire M_AXI_BREADY, //写响应信号接收有效,主到从
  102. // Master Interface Read Address.读地址ID
  103. output wire [C_M_AXI_ID_WIDTH-1 : 0] M_AXI_ARID,
  104. // Read address. This signal indicates the initial
  105. // address of a read burst transaction. 读地址
  106. output wire [C_M_AXI_ADDR_WIDTH-1 : 0] M_AXI_ARADDR,
  107. // Burst length. The burst length gives the exact number of transfers in a burst
  108. output wire [7 : 0] M_AXI_ARLEN, //读突发长度
  109. // Burst size. This signal indicates the size of each transfer in the burst
  110. output wire [2 : 0] M_AXI_ARSIZE, //读突发大小
  111. // Burst type. The burst type and the size information,
  112. // determine how the address for each transfer within the burst is calculated.
  113. output wire [1 : 0] M_AXI_ARBURST, //读突发类型
  114. // Lock type. Provides additional information about the
  115. // atomic characteristics of the transfer.
  116. output wire M_AXI_ARLOCK, //总线锁信号
  117. // Memory type. This signal indicates how transactions
  118. // are required to progress through a system.
  119. output wire [3 : 0] M_AXI_ARCACHE, //Cache类型,是否缓存
  120. // Protection type. This signal indicates the privilege
  121. // and security level of the transaction, and whether
  122. // the transaction is a data access or an instruction access.
  123. output wire [2 : 0] M_AXI_ARPROT, //保护类型
  124. // Quality of Service, QoS identifier sent for each read transaction
  125. output wire [3 : 0] M_AXI_ARQOS, //质量服务
  126. // Optional User-defined signal in the read address channel.
  127. output wire [C_M_AXI_ARUSER_WIDTH-1 : 0] M_AXI_ARUSER,
  128. // Write address valid. This signal indicates that
  129. // the channel is signaling valid read address and control information
  130. output wire M_AXI_ARVALID, //读地址发送有效,主到从
  131. // Read address ready. This signal indicates that
  132. // the slave is ready to accept an address and associated control signals
  133. input wire M_AXI_ARREADY, //读地址接收有效,从到主
  134. // Read ID tag. This signal is the identification tag
  135. // for the read data group of signals generated by the slave.
  136. input wire [C_M_AXI_ID_WIDTH-1 : 0] M_AXI_RID, //读数据ID
  137. // Master Read Data 读数据
  138. input wire [C_M_AXI_DATA_WIDTH-1 : 0] M_AXI_RDATA,
  139. // Read response. This signal indicates the status of the read transfer
  140. input wire [1 : 0] M_AXI_RRESP, //读响应信号
  141. // Read last. This signal indicates the last transfer in a read burst
  142. input wire M_AXI_RLAST, //突发读传输的最后一个数据
  143. // Optional User-defined signal in the read address channel.
  144. input wire [C_M_AXI_RUSER_WIDTH-1 : 0] M_AXI_RUSER, //用户自定义信号
  145. // Read valid. This signal indicates that the channel
  146. // is signaling the required read data.
  147. input wire M_AXI_RVALID, //读数据发送有效,从到主
  148. // Read ready. This signal indicates that the master can
  149. // accept the read data and response information.
  150. output wire M_AXI_RREADY //读数据接收有效。主到从
  151. );
  152. // function called clogb2 that returns an integer which has the
  153. //value of the ceiling of the log base 2
  154. // function called clogb2 that returns an integer which has the
  155. // value of the ceiling of the log base 2.
  156. function integer clogb2 (input integer bit_depth); //计算信号的位宽
  157. begin
  158. for(clogb2=0; bit_depth>0; clogb2=clogb2+1)
  159. bit_depth = bit_depth >> 1;
  160. end
  161. endfunction
  162. // C_TRANSACTIONS_NUM is the width of the index counter for
  163. // number of write or read transaction.(4) C_TRANSACTIONS_NUM指读/写标识计数器的位宽
  164. localparam integer C_TRANSACTIONS_NUM = clogb2(C_M_AXI_BURST_LEN-1);
  165. // Burst length for transactions, in C_M_AXI_DATA_WIDTHs.
  166. // Non-2^n lengths will eventually cause bursts across 4K address boundaries.
  167. localparam integer C_MASTER_LENGTH = 12;
  168. // total number of burst transfers is master length divided by burst length and burst size
  169. localparam integer C_NO_BURSTS_REQ = C_MASTER_LENGTH-clogb2((C_M_AXI_BURST_LEN*C_M_AXI_DATA_WIDTH/8)-1);//12-6 = 6
  170. // Example State machine to initialize counter, initialize write transactions,
  171. // initialize read transactions and comparison of read data with the
  172. // written data words.
  173. parameter [1:0] IDLE = 2'b00, // This state initiates AXI4Lite transaction
  174. // after the state machine changes state to INIT_WRITE
  175. // when there is 0 to 1 transition on INIT_AXI_TXN 上升沿
  176. INIT_WRITE = 2'b01, // This state initializes write transaction,
  177. // once writes are done, the state machine
  178. // changes state to INIT_READ
  179. INIT_READ = 2'b10, // This state initializes read transaction
  180. // once reads are done, the state machine
  181. // changes state to INIT_COMPARE
  182. INIT_COMPARE = 2'b11; // This state issues the status of comparison
  183. // of the written data with the read data
  184. reg [1:0] mst_exec_state;
  185. // AXI4LITE signals
  186. //AXI4 internal temp signals
  187. reg [C_M_AXI_ADDR_WIDTH-1 : 0] axi_awaddr; //32位写地址
  188. reg axi_awvalid;
  189. reg [C_M_AXI_DATA_WIDTH-1 : 0] axi_wdata; //32位写数据
  190. reg axi_wlast;
  191. reg axi_wvalid;
  192. reg axi_bready;
  193. reg [C_M_AXI_ADDR_WIDTH-1 : 0] axi_araddr; //32位读地址
  194. reg axi_arvalid;
  195. reg axi_rready;
  196. //write beat count in a burst 写计数5位
  197. reg [C_TRANSACTIONS_NUM : 0] write_index;
  198. //read beat count in a burst 读计数5位
  199. reg [C_TRANSACTIONS_NUM : 0] read_index;
  200. //size of C_M_AXI_BURST_LEN length burst in bytes 7位
  201. wire [C_TRANSACTIONS_NUM+2 : 0] burst_size_bytes;
  202. //The burst counters are used to track the number of burst transfers of C_M_AXI_BURST_LEN burst length needed to transfer 2^C_MASTER_LENGTH bytes of data.
  203. reg [C_NO_BURSTS_REQ : 0] write_burst_counter; //7位
  204. reg [C_NO_BURSTS_REQ : 0] read_burst_counter; //7位
  205. reg start_single_burst_write;
  206. reg start_single_burst_read;
  207. reg writes_done;
  208. reg reads_done;
  209. reg error_reg;
  210. reg compare_done;
  211. reg read_mismatch;
  212. reg burst_write_active;
  213. reg burst_read_active;
  214. reg [C_M_AXI_DATA_WIDTH-1 : 0] expected_rdata;
  215. //Interface response error flags
  216. wire write_resp_error;
  217. wire read_resp_error;
  218. wire wnext;
  219. wire rnext;
  220. reg init_txn_ff;
  221. reg init_txn_ff2;
  222. reg init_txn_edge;
  223. wire init_txn_pulse;
  224. // I/O Connections assignments
  225. wire douta;
  226. wire ena;
  227. //I/O Connections. Write Address (AW)
  228. assign M_AXI_AWID = 'b0;
  229. //The AXI address is a concatenation of the target base address + active offset range
  230. assign M_AXI_AWADDR = C_M_TARGET_SLAVE_BASE_ADDR + axi_awaddr;
  231. //Burst LENgth is number of transaction beats, minus 1
  232. assign M_AXI_AWLEN = C_M_AXI_BURST_LEN - 1;
  233. //Size should be C_M_AXI_DATA_WIDTH, in 2^SIZE bytes, otherwise narrow bursts are used
  234. assign M_AXI_AWSIZE = clogb2((C_M_AXI_DATA_WIDTH/8)-1);//2'b10; 4BYTE
  235. //INCR burst type is usually used, except for keyhole bursts
  236. assign M_AXI_AWBURST = 2'b01; //INCR自增类型
  237. assign M_AXI_AWLOCK = 1'b0;
  238. //Update value to 4'b0011 if coherent accesses to be used via the Zynq ACP port. Not Allocated, Modifiable, not Bufferable. Not Bufferable since this example is meant to test memory, not intermediate cache.
  239. assign M_AXI_AWCACHE = 4'b0010;
  240. assign M_AXI_AWPROT = 3'h0;
  241. assign M_AXI_AWQOS = 4'h0;
  242. assign M_AXI_AWUSER = 'b1;
  243. assign M_AXI_AWVALID = axi_awvalid;
  244. //Write Data(W)
  245. assign M_AXI_WDATA = douta;
  246. //All bursts are complete and aligned in this example
  247. assign M_AXI_WSTRB = {(C_M_AXI_DATA_WIDTH/8){1'b1}};//4'B1111;
  248. assign M_AXI_WLAST = axi_wlast;
  249. assign M_AXI_WUSER = 'b0;
  250. assign M_AXI_WVALID = axi_wvalid;
  251. //Write Response (B)
  252. assign M_AXI_BREADY = axi_bready;
  253. //Read Address (AR)
  254. assign M_AXI_ARID = 'b0;
  255. assign M_AXI_ARADDR = C_M_TARGET_SLAVE_BASE_ADDR + axi_araddr;
  256. //Burst LENgth is number of transaction beats, minus 1
  257. assign M_AXI_ARLEN = C_M_AXI_BURST_LEN - 1;
  258. //Size should be C_M_AXI_DATA_WIDTH, in 2^n bytes, otherwise narrow bursts are used
  259. assign M_AXI_ARSIZE = clogb2((C_M_AXI_DATA_WIDTH/8)-1);
  260. //INCR burst type is usually used, except for keyhole bursts
  261. assign M_AXI_ARBURST = 2'b01;
  262. assign M_AXI_ARLOCK = 1'b0;
  263. //Update value to 4'b0011 if coherent accesses to be used via the Zynq ACP port. Not Allocated, Modifiable, not Bufferable. Not Bufferable since this example is meant to test memory, not intermediate cache.
  264. assign M_AXI_ARCACHE = 4'b0010;
  265. assign M_AXI_ARPROT = 3'h0;
  266. assign M_AXI_ARQOS = 4'h0;
  267. assign M_AXI_ARUSER = 'b1;
  268. assign M_AXI_ARVALID = axi_arvalid;
  269. //Read and Read Response (R)
  270. assign M_AXI_RREADY = axi_rready;
  271. //Example design I/O
  272. assign TXN_DONE = compare_done;
  273. //Burst size in bytes
  274. assign burst_size_bytes = C_M_AXI_BURST_LEN * C_M_AXI_DATA_WIDTH/8;//16*32/8 = 64
  275. assign init_txn_pulse = (!init_txn_ff2) && init_txn_ff;//上升沿
  276. //Generate a pulse to initiate AXI transaction.
  277. always @(posedge M_AXI_ACLK) //对输入按键信号打两拍,消除亚稳态
  278. begin
  279. // Initiates AXI transaction delay
  280. if (M_AXI_ARESETN == 0 )
  281. begin
  282. init_txn_ff <= 1'b0;
  283. init_txn_ff2 <= 1'b0;
  284. end
  285. else
  286. begin
  287. init_txn_ff <= INIT_AXI_TXN;
  288. init_txn_ff2 <= init_txn_ff;
  289. end
  290. end
  291. //--------------------
  292. //Write Address Channel 写地址通道
  293. //--------------------
  294. // The purpose of the write address channel is to request the address and 写地址通道是传输整个传输过程中的首地址和控制信号
  295. // command information for the entire transaction. It is a single beat 只传输一次
  296. // of information.
  297. // The AXI4 Write address channel in this example will continue to initiate
  298. // write commands as fast as it is allowed by the slave/interconnect.
  299. // The address will be incremented on each accepted address transaction,
  300. // by burst_size_byte to point to the next address.
  301. always @(posedge M_AXI_ACLK)
  302. begin
  303. if (M_AXI_ARESETN == 0 || init_txn_pulse == 1'b1 )
  304. begin
  305. axi_awvalid <= 1'b0;
  306. end
  307. // If previously not valid , start next transaction
  308. else if (~axi_awvalid && start_single_burst_write)
  309. begin
  310. axi_awvalid <= 1'b1;
  311. end
  312. /* Once asserted, VALIDs cannot be deasserted, so axi_awvalid
  313. must wait until transaction is accepted */
  314. else if (M_AXI_AWREADY && axi_awvalid)
  315. begin
  316. axi_awvalid <= 1'b0;
  317. end
  318. else
  319. axi_awvalid <= axi_awvalid;
  320. end
  321. // Next address after AWREADY indicates previous address acceptance
  322. always @(posedge M_AXI_ACLK)
  323. begin
  324. if (M_AXI_ARESETN == 0 || init_txn_pulse == 1'b1)
  325. begin
  326. axi_awaddr <= 'b0;
  327. end
  328. else if (M_AXI_AWREADY && axi_awvalid)
  329. begin
  330. axi_awaddr <= axi_awaddr + burst_size_bytes; //突发长度为16,突发大小为4个字节,16*4=64
  331. end
  332. else
  333. axi_awaddr <= axi_awaddr;
  334. end
  335. //--------------------
  336. //Write Data Channel 写数据通道
  337. //--------------------
  338. //The write data will continually try to push write data across the interface.
  339. //The amount of data accepted will depend on the AXI slave and the AXI
  340. //Interconnect settings, such as if there are FIFOs enabled in interconnect.
  341. //Note that there is no explicit timing relationship to the write address channel.
  342. //The write channel has its own throttling flag, separate from the AW channel.
  343. //Synchronization between the channels must be determined by the user.
  344. //The simpliest but lowest performance would be to only issue one address write
  345. //and write data burst at a time.
  346. //In this example they are kept in sync by using the same address increment
  347. //and burst sizes. Then the AW and W channels have their transactions measured
  348. //with threshold counters as part of the user logic, to make sure neither
  349. //channel gets too far ahead of each other.
  350. //Forward movement occurs when the write channel is valid and ready
  351. assign wnext = M_AXI_WREADY & axi_wvalid; //wnext表示握手成功
  352. // WVALID logic, similar to the axi_awvalid always block above
  353. always @(posedge M_AXI_ACLK)
  354. begin
  355. if (M_AXI_ARESETN == 0 || init_txn_pulse == 1'b1 )
  356. begin
  357. axi_wvalid <= 1'b0;
  358. end
  359. // If previously not valid, start next transaction
  360. else if (~axi_wvalid && start_single_burst_write)
  361. begin
  362. axi_wvalid <= 1'b1;
  363. end
  364. /* If WREADY and too many writes, throttle WVALID
  365. Once asserted, VALIDs cannot be deasserted, so WVALID
  366. must wait until burst is complete with WLAST */
  367. else if (wnext && axi_wlast)
  368. axi_wvalid <= 1'b0;
  369. else
  370. axi_wvalid <= axi_wvalid;
  371. end

 9.然后大家在Block Design中添加刚刚生成的这个IP核,再添加一个DRAM,然后对模块进行连接。BRAM我的配置如下所示:

 

 

 

然后其他的不用修改,直接添加,添加完成后对模块进行连接,使用一个控制模块将他们连起来。连接完成后如图所示: 

 10.在完成上述工作后,我们便将AXI的IP核与BRAM连接起来了,通过AXI接口给BRAM发送数据,然后再将数据读出来,通过验证输出结果。TX_DONE表示发送成功,若得到错误的结果,会将erro_flag拉高。我们将这个设计打包成.v文件,具体操作如下。

 11.这样我们就得到了这个测试系统的顶层文件,然后我们对其编写测试激励,仿真验证。

  1. module tb_design_1();
  2. reg KEY;
  3. reg SCLK;
  4. reg SRST_N;
  5. wire TX_done;
  6. wire error_flag;
  7. wire rsta_busy_0;
  8. design_1 design_1(
  9. .KEY(KEY),
  10. .SCLK(SCLK),
  11. .SRST_N(SRST_N),
  12. .TX_done(TX_done),
  13. .error_flag(error_flag),
  14. .rsta_busy_0(rsta_busy_0)
  15. );
  16. always #10 SCLK = ~SCLK;
  17. initial begin
  18. KEY = 0;
  19. SCLK = 0;
  20. SRST_N = 0;
  21. #201;
  22. SRST_N = 1;
  23. #100;
  24. KEY = 1;
  25. #200;
  26. KEY = 0;
  27. #200000;
  28. $stop;
  29. end
  30. endmodule

 12.如图所示,前半部分是写数据,后半部分是读数据,最后可以看到发送完成信号TX_DONE信号拉高,而erro_flag没有拉高,我们也可以观察数据得到这个结果是正确的。

 13.这样我们便通过AXI4的IP核对BRAM读写验证了功能的正确性,大家有兴趣的话可以看一下正点原子ZYNQ系列的视频,讲了该IP核去读写PS端DDR3的实验。

        后续我还会总结一些关于DDR3相关的知识以及通过MIG去控制DDR3,实现该AXI4 IP核对DDR3进行读写的实验。

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/从前慢现在也慢/article/detail/999950
推荐阅读
  

闽ICP备14008679号