2 * @file x393_tasks_mcntrl_timing.vh
4 * @author Andrey Filippov
6 * @brief Simulation tasks for programming I/O delays and other timing
7 * parameters in the memory controller
9 * @copyright Copyright (c) 2015 Elphel, Inc.
13 * x393_tasks_mcntrl_timing.vh is free software; you can redistribute it and/or modify
14 * it under the terms of the GNU General Public License as published by
15 * the Free Software Foundation, either version 3 of the License, or
16 * (at your option) any later version.
18 * x393_tasks_mcntrl_timing.vh is distributed in the hope that it will be useful,
19 * but WITHOUT ANY WARRANTY; without even the implied warranty of
20 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
21 * GNU General Public License for more details.
23 * You should have received a copy of the GNU General Public License
24 * along with this program. If not, see <http://www.gnu.org/licenses/> .
26 * Additional permission under GNU GPL version 3 section 7:
27 * If you modify this Program, or any covered work, by linking or combining it
28 * with independent modules provided by the FPGA vendor only (this permission
29 * does not extend to any 3-rd party modules, "soft cores" or macros) under
30 * different license terms solely for the purpose of generating binary "bitstream"
31 * files and/or simulating the code, the copyright holders of this Program give
32 * you the right to distribute the covered work without those independent modules
33 * as long as the source code for them is available from the FPGA vendor free of
34 * charge, and there is no dependence on any encrypted modules for simulating of
35 * the combined code. This permission applies to you if the distributed code
36 * contains all the components and scripts required to completely simulate it
37 * with at least one of the Free Software programs.
40 task axi_set_same_delays; //SuppressThisWarning VEditor : may be unused
41 input [7:0] dq_idelay;
42 input [7:0] dq_odelay;
43 input [7:0] dqs_idelay;
44 input [7:0] dqs_odelay;
45 input [7:0] dm_odelay;
46 input [7:0] cmda_odelay;
48 $display("SET DELAYS(0x%x,0x%x,0x%x,0x%x,0x%x,0x%x) @ %t",
49 dq_idelay,dq_odelay,dqs_idelay,dqs_odelay,dm_odelay,cmda_odelay,$time);
50 axi_set_dq_idelay(dq_idelay);
51 axi_set_dq_odelay(dq_odelay);
52 axi_set_dqs_idelay(dqs_idelay);
53 axi_set_dqs_odelay(dqs_odelay);
54 axi_set_dm_odelay(dm_odelay);
55 axi_set_cmda_odelay(cmda_odelay);
59 task axi_set_dqs_odelay_nominal; //SuppressThisWarning VEditor : may be unused
61 $display("axi_set_dqs_odelay_nominal(0x%x,0x%x) @ %t",
62 (DLY_LANE0_ODELAY >> (8<<3)) & 32'hff,
63 (DLY_LANE1_ODELAY >> (8<<3)) & 32'hff,
65 // axi_set_dqs_idelay(
66 write_contol_register(LD_DLY_LANE0_ODELAY + 8, (DLY_LANE0_ODELAY >> (8<<3)) & 32'hff);
67 write_contol_register(LD_DLY_LANE1_ODELAY + 8, (DLY_LANE1_ODELAY >> (8<<3)) & 32'hff);
68 write_contol_register(DLY_SET,0);
72 task axi_set_dqs_idelay_nominal; //SuppressThisWarning VEditor : may be unused
74 // axi_set_dqs_idelay(
75 write_contol_register(LD_DLY_LANE0_IDELAY + 8, (DLY_LANE0_IDELAY >> (8<<3)) & 32'hff);
76 write_contol_register(LD_DLY_LANE1_IDELAY + 8, (DLY_LANE1_IDELAY >> (8<<3)) & 32'hff);
77 write_contol_register(DLY_SET,0);
81 task axi_set_dqs_idelay_wlv; //SuppressThisWarning VEditor : may be unused
83 write_contol_register(LD_DLY_LANE0_IDELAY + 8, DLY_LANE0_DQS_WLV_IDELAY);
84 write_contol_register(LD_DLY_LANE1_IDELAY + 8, DLY_LANE1_DQS_WLV_IDELAY);
85 write_contol_register(DLY_SET,0);
89 task axi_set_delays; // set all individual delays
92 $display("axi_set_delays @ %t",$time);
94 for (i=0;i<10;i=i+1) begin
95 write_contol_register(LD_DLY_LANE0_ODELAY + i, (DLY_LANE0_ODELAY >> (i<<3)) & 32'hff);
97 for (i=0;i<9;i=i+1) begin
98 write_contol_register(LD_DLY_LANE0_IDELAY + i, (DLY_LANE0_IDELAY >> (i<<3)) & 32'hff);
100 for (i=0;i<10;i=i+1) begin
101 write_contol_register(LD_DLY_LANE1_ODELAY + i, (DLY_LANE1_ODELAY >> (i<<3)) & 32'hff);
103 for (i=0;i<9;i=i+1) begin
104 write_contol_register(LD_DLY_LANE1_IDELAY + i, (DLY_LANE1_IDELAY >> (i<<3)) & 32'hff);
106 for (i=0;i<32;i=i+1) begin
107 write_contol_register(LD_DLY_CMDA + i, (DLY_CMDA >> (i<<3)) & 32'hff);
109 // write_contol_register(DLY_SET,0);
110 axi_set_phase(DLY_PHASE); // also sets all delays
114 task axi_get_delays; // set all individual delays
117 $display("axi_get_delays @ %t",$time);
119 for (i=0;i<10;i=i+1) begin
120 read_contol_register(LD_DLY_LANE0_ODELAY + i);
122 for (i=0;i<9;i=i+1) begin
123 read_contol_register(LD_DLY_LANE0_IDELAY + i);
125 for (i=0;i<10;i=i+1) begin
126 read_contol_register(LD_DLY_LANE1_ODELAY + i);
128 for (i=0;i<9;i=i+1) begin
129 read_contol_register(LD_DLY_LANE1_IDELAY + i);
131 for (i=0;i<32;i=i+1) begin
132 read_contol_register(LD_DLY_CMDA + i);
134 read_contol_register(LD_DLY_PHASE);
139 task axi_set_dq_idelay; // sets same delay to all dq idelay
142 $display("SET DQ IDELAY=0x%x @ %t",delay,$time);
143 axi_set_multiple_delays(LD_DLY_LANE0_IDELAY, 8, delay);
144 axi_set_multiple_delays(LD_DLY_LANE1_IDELAY, 8, delay);
145 write_contol_register(DLY_SET,0); // set all delays
149 task axi_set_dq_odelay;
152 $display("SET DQ ODELAY=0x%x @ %t",delay,$time);
153 axi_set_multiple_delays(LD_DLY_LANE0_ODELAY, 8, delay);
154 axi_set_multiple_delays(LD_DLY_LANE1_ODELAY, 8, delay);
155 write_contol_register(DLY_SET,0); // set all delays
159 task axi_set_dqs_idelay;
162 $display("SET DQS IDELAY=0x%x @ %t",delay,$time);
163 axi_set_multiple_delays(LD_DLY_LANE0_IDELAY + 8, 1, delay);
164 axi_set_multiple_delays(LD_DLY_LANE1_IDELAY + 8, 1, delay);
165 write_contol_register(DLY_SET,0); // set all delays
169 task axi_set_dqs_odelay;
172 $display("SET DQS ODELAY=0x%x @ %t",delay,$time);
173 axi_set_multiple_delays(LD_DLY_LANE0_ODELAY + 8, 1, delay);
174 axi_set_multiple_delays(LD_DLY_LANE1_ODELAY + 8, 1, delay);
175 write_contol_register(DLY_SET,0); // set all delays
179 task axi_set_dm_odelay;
182 $display("SET DQM IDELAY=0x%x @ %t",delay,$time);
183 axi_set_multiple_delays(LD_DLY_LANE0_ODELAY + 9, 1, delay);
184 axi_set_multiple_delays(LD_DLY_LANE1_ODELAY + 9, 1, delay);
185 write_contol_register(DLY_SET,0); // set all delays
189 task axi_set_cmda_odelay;
192 $display("SET COMMAND and ADDRESS ODELAY=0x%x @ %t",delay,$time);
193 axi_set_multiple_delays(LD_DLY_CMDA, 32, delay);
194 write_contol_register(DLY_SET,0); // set all delays
199 task axi_set_multiple_delays;
200 input [29:0] reg_addr;
201 input integer number;
205 for (i=0;i<number;i=i+1) begin
206 write_contol_register(reg_addr + i, {24'b0,delay}); // control regiter address
212 input [PHASE_WIDTH-1:0] phase;
213 reg [PHASE_WIDTH-1:0] inverted_phase;
215 if (CLKFBOUT_USE_FINE_PS) begin
216 inverted_phase = -phase;
217 $display("SET CLOCK INVERTED PHASE (0x%x) to 0x%x @ %t",phase,inverted_phase,$time);
218 write_contol_register(LD_DLY_PHASE, {{(32-PHASE_WIDTH){1'b0}},inverted_phase}); // control regiter address
219 target_phase <= inverted_phase;
221 $display("SET CLOCK PHASE to 0x%x @ %t",phase,$time);
222 write_contol_register(LD_DLY_PHASE, {{(32-PHASE_WIDTH){1'b0}},phase}); // control regiter address
223 target_phase <= phase;
225 write_contol_register(DLY_SET,0);
229 task axi_set_wbuf_delay;
232 $display("SET WBUF DELAY to 0x%x @ %t",delay,$time);
233 write_contol_register(MCONTR_PHY_16BIT_ADDR+MCONTR_PHY_16BIT_WBUF_DELAY, {28'h0, delay});
238 // set dq /dqs tristate on/off patterns
239 task axi_set_tristate_patterns;
241 $display("SET TRISTATE PATTERNS @ %t",$time);
242 write_contol_register(MCONTR_PHY_16BIT_ADDR +MCONTR_PHY_16BIT_PATTERNS_TRI,
243 {16'h0, DQSTRI_LAST, DQSTRI_FIRST, DQTRI_LAST, DQTRI_FIRST});
247 task axi_set_dqs_dqm_patterns;
249 $display("SET DQS+DQM PATTERNS @ %t",$time);
250 // set patterns for DM (always 0) and DQS - always the same (may try different for write lev.)
251 write_contol_register(MCONTR_PHY_16BIT_ADDR + MCONTR_PHY_16BIT_PATTERNS,