xref: /openbmc/u-boot/drivers/ddr/marvell/a38x/ddr3_training_leveling.c (revision 8cb8c0c6a83bef319023ac2e967a85e1e92e18bc)
183d290c5STom Rini // SPDX-License-Identifier: GPL-2.0
2f1df9364SStefan Roese /*
3f1df9364SStefan Roese  * Copyright (C) Marvell International Ltd. and its affiliates
4f1df9364SStefan Roese  */
5f1df9364SStefan Roese 
6f1df9364SStefan Roese #include "ddr3_init.h"
7*ebb1a593SChris Packham #include "mv_ddr_training_db.h"
8*ebb1a593SChris Packham #include "ddr_training_ip_db.h"
9*ebb1a593SChris Packham #include "mv_ddr_regs.h"
10f1df9364SStefan Roese 
11f1df9364SStefan Roese #define WL_ITERATION_NUM	10
12f1df9364SStefan Roese 
13f1df9364SStefan Roese static u32 pup_mask_table[] = {
14f1df9364SStefan Roese 	0x000000ff,
15f1df9364SStefan Roese 	0x0000ff00,
16f1df9364SStefan Roese 	0x00ff0000,
17f1df9364SStefan Roese 	0xff000000
18f1df9364SStefan Roese };
19f1df9364SStefan Roese 
20f1df9364SStefan Roese static struct write_supp_result wr_supp_res[MAX_INTERFACE_NUM][MAX_BUS_NUM];
21f1df9364SStefan Roese 
22f1df9364SStefan Roese static int ddr3_tip_dynamic_write_leveling_seq(u32 dev_num);
23f1df9364SStefan Roese static int ddr3_tip_dynamic_read_leveling_seq(u32 dev_num);
24f1df9364SStefan Roese static int ddr3_tip_dynamic_per_bit_read_leveling_seq(u32 dev_num);
25f1df9364SStefan Roese static int ddr3_tip_wl_supp_align_phase_shift(u32 dev_num, u32 if_id,
262b4ffbf6SChris Packham 					      u32 bus_id);
27f1df9364SStefan Roese static int ddr3_tip_xsb_compare_test(u32 dev_num, u32 if_id, u32 bus_id,
282b4ffbf6SChris Packham 				     u32 edge_offset);
29f1df9364SStefan Roese 
302b4ffbf6SChris Packham enum {
312b4ffbf6SChris Packham 	PASS,
322b4ffbf6SChris Packham 	FAIL
332b4ffbf6SChris Packham };
34f1df9364SStefan Roese /*****************************************************************************
35f1df9364SStefan Roese Dynamic read leveling
36f1df9364SStefan Roese ******************************************************************************/
ddr3_tip_dynamic_read_leveling(u32 dev_num,u32 freq)37f1df9364SStefan Roese int ddr3_tip_dynamic_read_leveling(u32 dev_num, u32 freq)
38f1df9364SStefan Roese {
39f1df9364SStefan Roese 	u32 data, mask;
40*ebb1a593SChris Packham 	unsigned int max_cs = mv_ddr_cs_num_get();
41f1df9364SStefan Roese 	u32 bus_num, if_id, cl_val;
42*ebb1a593SChris Packham 	enum mv_ddr_speed_bin speed_bin_index;
43f1df9364SStefan Roese 	/* save current CS value */
44f1df9364SStefan Roese 	u32 cs_enable_reg_val[MAX_INTERFACE_NUM] = { 0 };
45f1df9364SStefan Roese 	int is_any_pup_fail = 0;
46f1df9364SStefan Roese 	u32 data_read[MAX_INTERFACE_NUM + 1] = { 0 };
47*ebb1a593SChris Packham 	u8 rl_values[MAX_CS_NUM][MAX_BUS_NUM][MAX_INTERFACE_NUM];
48f1df9364SStefan Roese 	struct pattern_info *pattern_table = ddr3_tip_get_pattern_table();
49f1df9364SStefan Roese 	u16 *mask_results_pup_reg_map = ddr3_tip_get_mask_results_pup_reg_map();
502b4ffbf6SChris Packham 	u32 octets_per_if_num = ddr3_tip_dev_attr_get(dev_num, MV_ATTR_OCTET_PER_INTERFACE);
512b4ffbf6SChris Packham 	struct mv_ddr_topology_map *tm = mv_ddr_topology_map_get();
52f1df9364SStefan Roese 
53*ebb1a593SChris Packham 	for (effective_cs = 0; effective_cs < MAX_CS_NUM; effective_cs++)
54f1df9364SStefan Roese 		for (bus_num = 0; bus_num < MAX_BUS_NUM; bus_num++)
55f1df9364SStefan Roese 			for (if_id = 0; if_id <= MAX_INTERFACE_NUM - 1; if_id++)
56f1df9364SStefan Roese 				rl_values[effective_cs][bus_num][if_id] = 0;
57f1df9364SStefan Roese 
58f1df9364SStefan Roese 	for (effective_cs = 0; effective_cs < max_cs; effective_cs++) {
59f1df9364SStefan Roese 		for (if_id = 0; if_id <= MAX_INTERFACE_NUM - 1; if_id++) {
602b4ffbf6SChris Packham 			VALIDATE_IF_ACTIVE(tm->if_act_mask, if_id);
61f1df9364SStefan Roese 			training_result[training_stage][if_id] = TEST_SUCCESS;
62f1df9364SStefan Roese 
63f1df9364SStefan Roese 			/* save current cs enable reg val */
64f1df9364SStefan Roese 			CHECK_STATUS(ddr3_tip_if_read
65f1df9364SStefan Roese 				     (dev_num, ACCESS_TYPE_UNICAST, if_id,
662b4ffbf6SChris Packham 				      DUAL_DUNIT_CFG_REG, cs_enable_reg_val,
67f1df9364SStefan Roese 				      MASK_ALL_BITS));
68f1df9364SStefan Roese 			/* enable single cs */
69f1df9364SStefan Roese 			CHECK_STATUS(ddr3_tip_if_write
70f1df9364SStefan Roese 				     (dev_num, ACCESS_TYPE_UNICAST, if_id,
712b4ffbf6SChris Packham 				      DUAL_DUNIT_CFG_REG, (1 << 3), (1 << 3)));
72f1df9364SStefan Roese 		}
73f1df9364SStefan Roese 
74f1df9364SStefan Roese 		ddr3_tip_reset_fifo_ptr(dev_num);
75f1df9364SStefan Roese 
76f1df9364SStefan Roese 		/*
77f1df9364SStefan Roese 		 *     Phase 1: Load pattern (using ODPG)
78f1df9364SStefan Roese 		 *
79f1df9364SStefan Roese 		 * enter Read Leveling mode
80f1df9364SStefan Roese 		 * only 27 bits are masked
81f1df9364SStefan Roese 		 * assuming non multi-CS configuration
82f1df9364SStefan Roese 		 * write to CS = 0 for the non multi CS configuration, note
83f1df9364SStefan Roese 		 * that the results shall be read back to the required CS !!!
84f1df9364SStefan Roese 		 */
85f1df9364SStefan Roese 
86f1df9364SStefan Roese 		/* BUS count is 0 shifted 26 */
87f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
88f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
892b4ffbf6SChris Packham 			      ODPG_DATA_CTRL_REG, 0x3, 0x3));
90f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_configure_odpg
91f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE, 0,
92f1df9364SStefan Roese 			      pattern_table[PATTERN_RL].num_of_phases_tx, 0,
93f1df9364SStefan Roese 			      pattern_table[PATTERN_RL].num_of_phases_rx, 0, 0,
94f1df9364SStefan Roese 			      effective_cs, STRESS_NONE, DURATION_SINGLE));
95f1df9364SStefan Roese 
96f1df9364SStefan Roese 		/* load pattern to ODPG */
97f1df9364SStefan Roese 		ddr3_tip_load_pattern_to_odpg(dev_num, ACCESS_TYPE_MULTICAST,
98f1df9364SStefan Roese 					      PARAM_NOT_CARE, PATTERN_RL,
99f1df9364SStefan Roese 					      pattern_table[PATTERN_RL].
100f1df9364SStefan Roese 					      start_addr);
101f1df9364SStefan Roese 
102f1df9364SStefan Roese 		/*
103f1df9364SStefan Roese 		 *     Phase 2: ODPG to Read Leveling mode
104f1df9364SStefan Roese 		 */
105f1df9364SStefan Roese 
106f1df9364SStefan Roese 		/* General Training Opcode register */
107f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
108f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
1092b4ffbf6SChris Packham 			      ODPG_WR_RD_MODE_ENA_REG, 0,
110f1df9364SStefan Roese 			      MASK_ALL_BITS));
111f1df9364SStefan Roese 
112f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
113f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
1142b4ffbf6SChris Packham 			      GENERAL_TRAINING_OPCODE_REG,
115f1df9364SStefan Roese 			      (0x301b01 | effective_cs << 2), 0x3c3fef));
116f1df9364SStefan Roese 
117f1df9364SStefan Roese 		/* Object1 opcode register 0 & 1 */
118f1df9364SStefan Roese 		for (if_id = 0; if_id <= MAX_INTERFACE_NUM - 1; if_id++) {
1192b4ffbf6SChris Packham 			VALIDATE_IF_ACTIVE(tm->if_act_mask, if_id);
120f1df9364SStefan Roese 			speed_bin_index =
121f1df9364SStefan Roese 				tm->interface_params[if_id].speed_bin_index;
122*ebb1a593SChris Packham 			cl_val = mv_ddr_cl_val_get(speed_bin_index, freq);
123f1df9364SStefan Roese 			data = (cl_val << 17) | (0x3 << 25);
124f1df9364SStefan Roese 			mask = (0xff << 9) | (0x1f << 17) | (0x3 << 25);
125f1df9364SStefan Roese 			CHECK_STATUS(ddr3_tip_if_write
126f1df9364SStefan Roese 				     (dev_num, ACCESS_TYPE_UNICAST, if_id,
1272b4ffbf6SChris Packham 				      OPCODE_REG0_REG(1), data, mask));
128f1df9364SStefan Roese 		}
129f1df9364SStefan Roese 
130f1df9364SStefan Roese 		/* Set iteration count to max value */
131f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
132f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
1332b4ffbf6SChris Packham 			      OPCODE_REG1_REG(1), 0xd00, 0xd00));
134f1df9364SStefan Roese 
135f1df9364SStefan Roese 		/*
136f1df9364SStefan Roese 		 *     Phase 2: Mask config
137f1df9364SStefan Roese 		 */
138f1df9364SStefan Roese 
139f1df9364SStefan Roese 		ddr3_tip_dynamic_read_leveling_seq(dev_num);
140f1df9364SStefan Roese 
141f1df9364SStefan Roese 		/*
142f1df9364SStefan Roese 		 *     Phase 3: Read Leveling execution
143f1df9364SStefan Roese 		 */
144f1df9364SStefan Roese 
145f1df9364SStefan Roese 		/* temporary jira dunit=14751 */
146f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
147f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
148f1df9364SStefan Roese 			      TRAINING_DBG_1_REG, 0, (u32)(1 << 31)));
149f1df9364SStefan Roese 		/* configure phy reset value */
150f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
151f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
152f1df9364SStefan Roese 			      TRAINING_DBG_3_REG, (0x7f << 24),
153f1df9364SStefan Roese 			      (u32)(0xff << 24)));
154f1df9364SStefan Roese 		/* data pup rd reset enable  */
155f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
156f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
1572b4ffbf6SChris Packham 			      SDRAM_CFG_REG, 0, (1 << 30)));
158f1df9364SStefan Roese 		/* data pup rd reset disable */
159f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
160f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
1612b4ffbf6SChris Packham 			      SDRAM_CFG_REG, (1 << 30), (1 << 30)));
162f1df9364SStefan Roese 		/* training SW override & training RL mode */
163f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
164f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
165f1df9364SStefan Roese 			      TRAINING_SW_2_REG, 0x1, 0x9));
166f1df9364SStefan Roese 		/* training enable */
167f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
168f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
169f1df9364SStefan Roese 			      TRAINING_REG, (1 << 24) | (1 << 20),
170f1df9364SStefan Roese 			      (1 << 24) | (1 << 20)));
171f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
172f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
173f1df9364SStefan Roese 			      TRAINING_REG, (u32)(1 << 31), (u32)(1 << 31)));
174f1df9364SStefan Roese 
1752b4ffbf6SChris Packham 		/* trigger training */
1762b4ffbf6SChris Packham 		mv_ddr_training_enable();
177f1df9364SStefan Roese 
1782b4ffbf6SChris Packham 		/* check for training done */
1792b4ffbf6SChris Packham 		if (mv_ddr_is_training_done(MAX_POLLING_ITERATIONS, &data) != MV_OK) {
1802b4ffbf6SChris Packham 			DEBUG_LEVELING(DEBUG_LEVEL_ERROR, ("training done failed\n"));
1812b4ffbf6SChris Packham 			return MV_FAIL;
1822b4ffbf6SChris Packham 		}
1832b4ffbf6SChris Packham 		/* check for training pass */
1842b4ffbf6SChris Packham 		if (data != PASS)
1852b4ffbf6SChris Packham 			DEBUG_LEVELING(DEBUG_LEVEL_INFO, ("training result failed\n"));
1862b4ffbf6SChris Packham 
1872b4ffbf6SChris Packham 		/* disable odpg; switch back to functional mode */
1882b4ffbf6SChris Packham 		mv_ddr_odpg_disable();
1892b4ffbf6SChris Packham 
1902b4ffbf6SChris Packham 		if (mv_ddr_is_odpg_done(MAX_POLLING_ITERATIONS) != MV_OK) {
1912b4ffbf6SChris Packham 			DEBUG_LEVELING(DEBUG_LEVEL_ERROR, ("odpg disable failed\n"));
192f1df9364SStefan Roese 			return MV_FAIL;
193f1df9364SStefan Roese 		}
194f1df9364SStefan Roese 
1952b4ffbf6SChris Packham 		ddr3_tip_if_write(0, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
1962b4ffbf6SChris Packham 				  ODPG_DATA_CTRL_REG, 0, MASK_ALL_BITS);
197f1df9364SStefan Roese 
198f1df9364SStefan Roese 		/* double loop on bus, pup */
199f1df9364SStefan Roese 		for (if_id = 0; if_id <= MAX_INTERFACE_NUM - 1; if_id++) {
2002b4ffbf6SChris Packham 			VALIDATE_IF_ACTIVE(tm->if_act_mask, if_id);
201f1df9364SStefan Roese 			/* check training done */
202f1df9364SStefan Roese 			is_any_pup_fail = 0;
203f1df9364SStefan Roese 			for (bus_num = 0;
2042b4ffbf6SChris Packham 			     bus_num < octets_per_if_num;
205f1df9364SStefan Roese 			     bus_num++) {
2062b4ffbf6SChris Packham 				VALIDATE_BUS_ACTIVE(tm->bus_act_mask, bus_num);
207f1df9364SStefan Roese 				if (ddr3_tip_if_polling
208f1df9364SStefan Roese 				    (dev_num, ACCESS_TYPE_UNICAST,
209f1df9364SStefan Roese 				     if_id, (1 << 25), (1 << 25),
210f1df9364SStefan Roese 				     mask_results_pup_reg_map[bus_num],
211f1df9364SStefan Roese 				     MAX_POLLING_ITERATIONS) != MV_OK) {
212f1df9364SStefan Roese 					DEBUG_LEVELING(DEBUG_LEVEL_ERROR,
2132b4ffbf6SChris Packham 						       ("\n_r_l: DDR3 poll failed(2) for IF %d CS %d bus %d",
2142b4ffbf6SChris Packham 							if_id, effective_cs, bus_num));
215f1df9364SStefan Roese 					is_any_pup_fail = 1;
216f1df9364SStefan Roese 				} else {
217f1df9364SStefan Roese 					/* read result per pup */
218f1df9364SStefan Roese 					CHECK_STATUS(ddr3_tip_if_read
219f1df9364SStefan Roese 						     (dev_num,
220f1df9364SStefan Roese 						      ACCESS_TYPE_UNICAST,
221f1df9364SStefan Roese 						      if_id,
222f1df9364SStefan Roese 						      mask_results_pup_reg_map
223f1df9364SStefan Roese 						      [bus_num], data_read,
224f1df9364SStefan Roese 						      0xff));
225f1df9364SStefan Roese 					rl_values[effective_cs][bus_num]
226f1df9364SStefan Roese 						[if_id] = (u8)data_read[if_id];
227f1df9364SStefan Roese 				}
228f1df9364SStefan Roese 			}
229f1df9364SStefan Roese 
230f1df9364SStefan Roese 			if (is_any_pup_fail == 1) {
231f1df9364SStefan Roese 				training_result[training_stage][if_id] =
232f1df9364SStefan Roese 					TEST_FAILED;
233f1df9364SStefan Roese 				if (debug_mode == 0)
234f1df9364SStefan Roese 					return MV_FAIL;
235f1df9364SStefan Roese 			}
236f1df9364SStefan Roese 		}
237f1df9364SStefan Roese 
238f1df9364SStefan Roese 		DEBUG_LEVELING(DEBUG_LEVEL_INFO, ("RL exit read leveling\n"));
239f1df9364SStefan Roese 
240f1df9364SStefan Roese 		/*
241f1df9364SStefan Roese 		 *     Phase 3: Exit Read Leveling
242f1df9364SStefan Roese 		 */
243f1df9364SStefan Roese 
244f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
245f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
246f1df9364SStefan Roese 			      TRAINING_SW_2_REG, (1 << 3), (1 << 3)));
247f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
248f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
249f1df9364SStefan Roese 			      TRAINING_SW_1_REG, (1 << 16), (1 << 16)));
250f1df9364SStefan Roese 		/* set ODPG to functional */
251f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
252f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
2532b4ffbf6SChris Packham 			      ODPG_DATA_CTRL_REG, 0x0, MASK_ALL_BITS));
254f1df9364SStefan Roese 
255f1df9364SStefan Roese 		/*
256f1df9364SStefan Roese 		 * Copy the result from the effective CS search to the
257f1df9364SStefan Roese 		 * real Functional CS
258f1df9364SStefan Roese 		 */
2592b4ffbf6SChris Packham 		/*ddr3_tip_write_cs_result(dev_num, RL_PHY_REG(0); */
260f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
261f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
2622b4ffbf6SChris Packham 			      ODPG_DATA_CTRL_REG, 0x0, MASK_ALL_BITS));
263f1df9364SStefan Roese 	}
264f1df9364SStefan Roese 
265f1df9364SStefan Roese 	for (effective_cs = 0; effective_cs < max_cs; effective_cs++) {
266f1df9364SStefan Roese 		/* double loop on bus, pup */
267f1df9364SStefan Roese 		for (if_id = 0; if_id <= MAX_INTERFACE_NUM - 1; if_id++) {
2682b4ffbf6SChris Packham 			VALIDATE_IF_ACTIVE(tm->if_act_mask, if_id);
269f1df9364SStefan Roese 			for (bus_num = 0;
2702b4ffbf6SChris Packham 			     bus_num < octets_per_if_num;
271f1df9364SStefan Roese 			     bus_num++) {
2722b4ffbf6SChris Packham 				VALIDATE_BUS_ACTIVE(tm->bus_act_mask, bus_num);
273f1df9364SStefan Roese 				/* read result per pup from arry */
274f1df9364SStefan Roese 				data = rl_values[effective_cs][bus_num][if_id];
275f1df9364SStefan Roese 				data = (data & 0x1f) |
276f1df9364SStefan Roese 					(((data & 0xe0) >> 5) << 6);
277f1df9364SStefan Roese 				ddr3_tip_bus_write(dev_num,
278f1df9364SStefan Roese 						   ACCESS_TYPE_UNICAST,
279f1df9364SStefan Roese 						   if_id,
280f1df9364SStefan Roese 						   ACCESS_TYPE_UNICAST,
281f1df9364SStefan Roese 						   bus_num, DDR_PHY_DATA,
2822b4ffbf6SChris Packham 						   RL_PHY_REG(effective_cs),
2832b4ffbf6SChris Packham 						   data);
284f1df9364SStefan Roese 			}
285f1df9364SStefan Roese 		}
286f1df9364SStefan Roese 	}
287f1df9364SStefan Roese 	/* Set to 0 after each loop to avoid illegal value may be used */
288f1df9364SStefan Roese 	effective_cs = 0;
289f1df9364SStefan Roese 
290f1df9364SStefan Roese 	for (if_id = 0; if_id <= MAX_INTERFACE_NUM - 1; if_id++) {
2912b4ffbf6SChris Packham 		VALIDATE_IF_ACTIVE(tm->if_act_mask, if_id);
292f1df9364SStefan Roese 		/* restore cs enable value */
293f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
294f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_UNICAST, if_id,
2952b4ffbf6SChris Packham 			      DUAL_DUNIT_CFG_REG, cs_enable_reg_val[if_id],
296f1df9364SStefan Roese 			      MASK_ALL_BITS));
297f1df9364SStefan Roese 		if (odt_config != 0) {
298f1df9364SStefan Roese 			CHECK_STATUS(ddr3_tip_write_additional_odt_setting
299f1df9364SStefan Roese 				     (dev_num, if_id));
300f1df9364SStefan Roese 		}
301f1df9364SStefan Roese 	}
302f1df9364SStefan Roese 
303f1df9364SStefan Roese 	for (if_id = 0; if_id <= MAX_INTERFACE_NUM - 1; if_id++) {
3042b4ffbf6SChris Packham 		VALIDATE_IF_ACTIVE(tm->if_act_mask, if_id);
305f1df9364SStefan Roese 		if (training_result[training_stage][if_id] == TEST_FAILED)
306f1df9364SStefan Roese 			return MV_FAIL;
307f1df9364SStefan Roese 	}
308f1df9364SStefan Roese 
309f1df9364SStefan Roese 	return MV_OK;
310f1df9364SStefan Roese }
311f1df9364SStefan Roese 
312f1df9364SStefan Roese /*
313f1df9364SStefan Roese  * Legacy Dynamic write leveling
314f1df9364SStefan Roese  */
ddr3_tip_legacy_dynamic_write_leveling(u32 dev_num)315f1df9364SStefan Roese int ddr3_tip_legacy_dynamic_write_leveling(u32 dev_num)
316f1df9364SStefan Roese {
317f1df9364SStefan Roese 	u32 c_cs, if_id, cs_mask = 0;
318*ebb1a593SChris Packham 	unsigned int max_cs = mv_ddr_cs_num_get();
3192b4ffbf6SChris Packham 	struct mv_ddr_topology_map *tm = mv_ddr_topology_map_get();
320f1df9364SStefan Roese 
321f1df9364SStefan Roese 	/*
322f1df9364SStefan Roese 	 * In TRAINIUNG reg (0x15b0) write 0x80000008 | cs_mask:
323f1df9364SStefan Roese 	 * Trn_start
324f1df9364SStefan Roese 	 * cs_mask = 0x1 <<20 Trn_CS0 - CS0 is included in the DDR3 training
325f1df9364SStefan Roese 	 * cs_mask = 0x1 <<21 Trn_CS1 - CS1 is included in the DDR3 training
326f1df9364SStefan Roese 	 * cs_mask = 0x1 <<22 Trn_CS2 - CS2 is included in the DDR3 training
327f1df9364SStefan Roese 	 * cs_mask = 0x1 <<23 Trn_CS3 - CS3 is included in the DDR3 training
328f1df9364SStefan Roese 	 * Trn_auto_seq =  write leveling
329f1df9364SStefan Roese 	 */
330f1df9364SStefan Roese 	for (c_cs = 0; c_cs < max_cs; c_cs++)
331f1df9364SStefan Roese 		cs_mask = cs_mask | 1 << (20 + c_cs);
332f1df9364SStefan Roese 
333f1df9364SStefan Roese 	for (if_id = 0; if_id < MAX_INTERFACE_NUM; if_id++) {
3342b4ffbf6SChris Packham 		VALIDATE_IF_ACTIVE(tm->if_act_mask, if_id);
335f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
336f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_MULTICAST, 0,
337f1df9364SStefan Roese 			      TRAINING_REG, (0x80000008 | cs_mask),
338f1df9364SStefan Roese 			      0xffffffff));
339f1df9364SStefan Roese 		mdelay(20);
340f1df9364SStefan Roese 		if (ddr3_tip_if_polling
341f1df9364SStefan Roese 		    (dev_num, ACCESS_TYPE_UNICAST, if_id, 0,
342f1df9364SStefan Roese 		     (u32)0x80000000, TRAINING_REG,
343f1df9364SStefan Roese 		     MAX_POLLING_ITERATIONS) != MV_OK) {
344f1df9364SStefan Roese 			DEBUG_LEVELING(DEBUG_LEVEL_ERROR,
345f1df9364SStefan Roese 				       ("polling failed for Old WL result\n"));
346f1df9364SStefan Roese 			return MV_FAIL;
347f1df9364SStefan Roese 		}
348f1df9364SStefan Roese 	}
349f1df9364SStefan Roese 
350f1df9364SStefan Roese 	return MV_OK;
351f1df9364SStefan Roese }
352f1df9364SStefan Roese 
353f1df9364SStefan Roese /*
354f1df9364SStefan Roese  * Legacy Dynamic read leveling
355f1df9364SStefan Roese  */
ddr3_tip_legacy_dynamic_read_leveling(u32 dev_num)356f1df9364SStefan Roese int ddr3_tip_legacy_dynamic_read_leveling(u32 dev_num)
357f1df9364SStefan Roese {
358f1df9364SStefan Roese 	u32 c_cs, if_id, cs_mask = 0;
359*ebb1a593SChris Packham 	unsigned int max_cs = mv_ddr_cs_num_get();
3602b4ffbf6SChris Packham 	struct mv_ddr_topology_map *tm = mv_ddr_topology_map_get();
361f1df9364SStefan Roese 
362f1df9364SStefan Roese 	/*
363f1df9364SStefan Roese 	 * In TRAINIUNG reg (0x15b0) write 0x80000040 | cs_mask:
364f1df9364SStefan Roese 	 * Trn_start
365f1df9364SStefan Roese 	 * cs_mask = 0x1 <<20 Trn_CS0 - CS0 is included in the DDR3 training
366f1df9364SStefan Roese 	 * cs_mask = 0x1 <<21 Trn_CS1 - CS1 is included in the DDR3 training
367f1df9364SStefan Roese 	 * cs_mask = 0x1 <<22 Trn_CS2 - CS2 is included in the DDR3 training
368f1df9364SStefan Roese 	 * cs_mask = 0x1 <<23 Trn_CS3 - CS3 is included in the DDR3 training
369f1df9364SStefan Roese 	 * Trn_auto_seq =  Read Leveling using training pattern
370f1df9364SStefan Roese 	 */
371f1df9364SStefan Roese 	for (c_cs = 0; c_cs < max_cs; c_cs++)
372f1df9364SStefan Roese 		cs_mask = cs_mask | 1 << (20 + c_cs);
373f1df9364SStefan Roese 
374f1df9364SStefan Roese 	CHECK_STATUS(ddr3_tip_if_write
375f1df9364SStefan Roese 		     (dev_num, ACCESS_TYPE_MULTICAST, 0, TRAINING_REG,
376f1df9364SStefan Roese 		      (0x80000040 | cs_mask), 0xffffffff));
377f1df9364SStefan Roese 	mdelay(100);
378f1df9364SStefan Roese 
379f1df9364SStefan Roese 	for (if_id = 0; if_id < MAX_INTERFACE_NUM; if_id++) {
3802b4ffbf6SChris Packham 		VALIDATE_IF_ACTIVE(tm->if_act_mask, if_id);
381f1df9364SStefan Roese 		if (ddr3_tip_if_polling
382f1df9364SStefan Roese 		    (dev_num, ACCESS_TYPE_UNICAST, if_id, 0,
383f1df9364SStefan Roese 		     (u32)0x80000000, TRAINING_REG,
384f1df9364SStefan Roese 		     MAX_POLLING_ITERATIONS) != MV_OK) {
385f1df9364SStefan Roese 			DEBUG_LEVELING(DEBUG_LEVEL_ERROR,
386f1df9364SStefan Roese 				       ("polling failed for Old RL result\n"));
387f1df9364SStefan Roese 			return MV_FAIL;
388f1df9364SStefan Roese 		}
389f1df9364SStefan Roese 	}
390f1df9364SStefan Roese 
391f1df9364SStefan Roese 	return MV_OK;
392f1df9364SStefan Roese }
393f1df9364SStefan Roese 
394f1df9364SStefan Roese /*
395f1df9364SStefan Roese  * Dynamic per bit read leveling
396f1df9364SStefan Roese  */
ddr3_tip_dynamic_per_bit_read_leveling(u32 dev_num,u32 freq)397f1df9364SStefan Roese int ddr3_tip_dynamic_per_bit_read_leveling(u32 dev_num, u32 freq)
398f1df9364SStefan Roese {
399f1df9364SStefan Roese 	u32 data, mask;
400f1df9364SStefan Roese 	u32 bus_num, if_id, cl_val, bit_num;
401f1df9364SStefan Roese 	u32 curr_numb, curr_min_delay;
402f1df9364SStefan Roese 	int adll_array[3] = { 0, -0xa, 0x14 };
403f1df9364SStefan Roese 	u32 phyreg3_arr[MAX_INTERFACE_NUM][MAX_BUS_NUM];
404*ebb1a593SChris Packham 	enum mv_ddr_speed_bin speed_bin_index;
405f1df9364SStefan Roese 	int is_any_pup_fail = 0;
406f1df9364SStefan Roese 	int break_loop = 0;
407f1df9364SStefan Roese 	u32 cs_enable_reg_val[MAX_INTERFACE_NUM]; /* save current CS value */
408f1df9364SStefan Roese 	u32 data_read[MAX_INTERFACE_NUM];
409f1df9364SStefan Roese 	int per_bit_rl_pup_status[MAX_INTERFACE_NUM][MAX_BUS_NUM];
410f1df9364SStefan Roese 	u32 data2_write[MAX_INTERFACE_NUM][MAX_BUS_NUM];
411f1df9364SStefan Roese 	struct pattern_info *pattern_table = ddr3_tip_get_pattern_table();
412f1df9364SStefan Roese 	u16 *mask_results_dq_reg_map = ddr3_tip_get_mask_results_dq_reg();
4132b4ffbf6SChris Packham 	u32 octets_per_if_num = ddr3_tip_dev_attr_get(dev_num, MV_ATTR_OCTET_PER_INTERFACE);
4142b4ffbf6SChris Packham 	struct mv_ddr_topology_map *tm = mv_ddr_topology_map_get();
415f1df9364SStefan Roese 
416f1df9364SStefan Roese 	for (if_id = 0; if_id < MAX_INTERFACE_NUM; if_id++) {
4172b4ffbf6SChris Packham 		VALIDATE_IF_ACTIVE(tm->if_act_mask, if_id);
418f1df9364SStefan Roese 		for (bus_num = 0;
4192b4ffbf6SChris Packham 		     bus_num <= octets_per_if_num; bus_num++) {
4202b4ffbf6SChris Packham 			VALIDATE_BUS_ACTIVE(tm->bus_act_mask, bus_num);
421f1df9364SStefan Roese 			per_bit_rl_pup_status[if_id][bus_num] = 0;
422f1df9364SStefan Roese 			data2_write[if_id][bus_num] = 0;
423f1df9364SStefan Roese 			/* read current value of phy register 0x3 */
424f1df9364SStefan Roese 			CHECK_STATUS(ddr3_tip_bus_read
425f1df9364SStefan Roese 				     (dev_num, if_id, ACCESS_TYPE_UNICAST,
426f1df9364SStefan Roese 				      bus_num, DDR_PHY_DATA,
4272b4ffbf6SChris Packham 				      CRX_PHY_REG(0),
428f1df9364SStefan Roese 				      &phyreg3_arr[if_id][bus_num]));
429f1df9364SStefan Roese 		}
430f1df9364SStefan Roese 	}
431f1df9364SStefan Roese 
432f1df9364SStefan Roese 	/* NEW RL machine */
433f1df9364SStefan Roese 	for (if_id = 0; if_id <= MAX_INTERFACE_NUM - 1; if_id++) {
4342b4ffbf6SChris Packham 		VALIDATE_IF_ACTIVE(tm->if_act_mask, if_id);
435f1df9364SStefan Roese 		training_result[training_stage][if_id] = TEST_SUCCESS;
436f1df9364SStefan Roese 
437f1df9364SStefan Roese 		/* save current cs enable reg val */
438f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_read
439f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_UNICAST, if_id,
4402b4ffbf6SChris Packham 			      DUAL_DUNIT_CFG_REG, &cs_enable_reg_val[if_id],
441f1df9364SStefan Roese 			      MASK_ALL_BITS));
442f1df9364SStefan Roese 		/* enable single cs */
443f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
444f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_UNICAST, if_id,
4452b4ffbf6SChris Packham 			      DUAL_DUNIT_CFG_REG, (1 << 3), (1 << 3)));
446f1df9364SStefan Roese 	}
447f1df9364SStefan Roese 
448f1df9364SStefan Roese 	ddr3_tip_reset_fifo_ptr(dev_num);
449f1df9364SStefan Roese 	for (curr_numb = 0; curr_numb < 3; curr_numb++) {
450f1df9364SStefan Roese 		/*
451f1df9364SStefan Roese 		 *     Phase 1: Load pattern (using ODPG)
452f1df9364SStefan Roese 		 *
453f1df9364SStefan Roese 		 * enter Read Leveling mode
454f1df9364SStefan Roese 		 * only 27 bits are masked
455f1df9364SStefan Roese 		 * assuming non multi-CS configuration
456f1df9364SStefan Roese 		 * write to CS = 0 for the non multi CS configuration, note that
457f1df9364SStefan Roese 		 * the results shall be read back to the required CS !!!
458f1df9364SStefan Roese 		 */
459f1df9364SStefan Roese 
460f1df9364SStefan Roese 		/* BUS count is 0 shifted 26 */
461f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
462f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
4632b4ffbf6SChris Packham 			      ODPG_DATA_CTRL_REG, 0x3, 0x3));
464f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_configure_odpg
465f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE, 0,
466f1df9364SStefan Roese 			      pattern_table[PATTERN_TEST].num_of_phases_tx, 0,
467f1df9364SStefan Roese 			      pattern_table[PATTERN_TEST].num_of_phases_rx, 0,
468f1df9364SStefan Roese 			      0, 0, STRESS_NONE, DURATION_SINGLE));
469f1df9364SStefan Roese 
470f1df9364SStefan Roese 		/* load pattern to ODPG */
471f1df9364SStefan Roese 		ddr3_tip_load_pattern_to_odpg(dev_num, ACCESS_TYPE_MULTICAST,
472f1df9364SStefan Roese 					      PARAM_NOT_CARE, PATTERN_TEST,
473f1df9364SStefan Roese 					      pattern_table[PATTERN_TEST].
474f1df9364SStefan Roese 					      start_addr);
475f1df9364SStefan Roese 
476f1df9364SStefan Roese 		/*
477f1df9364SStefan Roese 		 *     Phase 2: ODPG to Read Leveling mode
478f1df9364SStefan Roese 		 */
479f1df9364SStefan Roese 
480f1df9364SStefan Roese 		/* General Training Opcode register */
481f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
482f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
4832b4ffbf6SChris Packham 			      ODPG_WR_RD_MODE_ENA_REG, 0,
484f1df9364SStefan Roese 			      MASK_ALL_BITS));
485f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
486f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
4872b4ffbf6SChris Packham 			      GENERAL_TRAINING_OPCODE_REG, 0x301b01, 0x3c3fef));
488f1df9364SStefan Roese 
489f1df9364SStefan Roese 		/* Object1 opcode register 0 & 1 */
490f1df9364SStefan Roese 		for (if_id = 0; if_id <= MAX_INTERFACE_NUM - 1; if_id++) {
4912b4ffbf6SChris Packham 			VALIDATE_IF_ACTIVE(tm->if_act_mask, if_id);
492f1df9364SStefan Roese 			speed_bin_index =
493f1df9364SStefan Roese 				tm->interface_params[if_id].speed_bin_index;
494*ebb1a593SChris Packham 			cl_val = mv_ddr_cl_val_get(speed_bin_index, freq);
495f1df9364SStefan Roese 			data = (cl_val << 17) | (0x3 << 25);
496f1df9364SStefan Roese 			mask = (0xff << 9) | (0x1f << 17) | (0x3 << 25);
497f1df9364SStefan Roese 			CHECK_STATUS(ddr3_tip_if_write
498f1df9364SStefan Roese 				     (dev_num, ACCESS_TYPE_UNICAST, if_id,
4992b4ffbf6SChris Packham 				      OPCODE_REG0_REG(1), data, mask));
500f1df9364SStefan Roese 		}
501f1df9364SStefan Roese 
502f1df9364SStefan Roese 		/* Set iteration count to max value */
503f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
504f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
5052b4ffbf6SChris Packham 			      OPCODE_REG1_REG(1), 0xd00, 0xd00));
506f1df9364SStefan Roese 
507f1df9364SStefan Roese 		/*
508f1df9364SStefan Roese 		 *     Phase 2: Mask config
509f1df9364SStefan Roese 		 */
510f1df9364SStefan Roese 
511f1df9364SStefan Roese 		ddr3_tip_dynamic_per_bit_read_leveling_seq(dev_num);
512f1df9364SStefan Roese 
513f1df9364SStefan Roese 		/*
514f1df9364SStefan Roese 		 *     Phase 3: Read Leveling execution
515f1df9364SStefan Roese 		 */
516f1df9364SStefan Roese 
517f1df9364SStefan Roese 		/* temporary jira dunit=14751 */
518f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
519f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
520f1df9364SStefan Roese 			      TRAINING_DBG_1_REG, 0, (u32)(1 << 31)));
521f1df9364SStefan Roese 		/* configure phy reset value */
522f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
523f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
524f1df9364SStefan Roese 			      TRAINING_DBG_3_REG, (0x7f << 24),
525f1df9364SStefan Roese 			      (u32)(0xff << 24)));
526f1df9364SStefan Roese 		/* data pup rd reset enable  */
527f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
528f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
5292b4ffbf6SChris Packham 			      SDRAM_CFG_REG, 0, (1 << 30)));
530f1df9364SStefan Roese 		/* data pup rd reset disable */
531f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
532f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
5332b4ffbf6SChris Packham 			      SDRAM_CFG_REG, (1 << 30), (1 << 30)));
534f1df9364SStefan Roese 		/* training SW override & training RL mode */
535f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
536f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
537f1df9364SStefan Roese 			      TRAINING_SW_2_REG, 0x1, 0x9));
538f1df9364SStefan Roese 		/* training enable */
539f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
540f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
541f1df9364SStefan Roese 			      TRAINING_REG, (1 << 24) | (1 << 20),
542f1df9364SStefan Roese 			      (1 << 24) | (1 << 20)));
543f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
544f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
545f1df9364SStefan Roese 			      TRAINING_REG, (u32)(1 << 31), (u32)(1 << 31)));
546f1df9364SStefan Roese 
5472b4ffbf6SChris Packham 		/* trigger training */
5482b4ffbf6SChris Packham 		mv_ddr_training_enable();
549f1df9364SStefan Roese 
5502b4ffbf6SChris Packham 		/* check for training done */
5512b4ffbf6SChris Packham 		if (mv_ddr_is_training_done(MAX_POLLING_ITERATIONS, &data) != MV_OK) {
5522b4ffbf6SChris Packham 			DEBUG_LEVELING(DEBUG_LEVEL_ERROR, ("training done failed\n"));
5532b4ffbf6SChris Packham 			return MV_FAIL;
5542b4ffbf6SChris Packham 		}
5552b4ffbf6SChris Packham 		/* check for training pass */
5562b4ffbf6SChris Packham 		if (data != PASS)
5572b4ffbf6SChris Packham 			DEBUG_LEVELING(DEBUG_LEVEL_INFO, ("training result failed\n"));
5582b4ffbf6SChris Packham 
5592b4ffbf6SChris Packham 		/* disable odpg; switch back to functional mode */
5602b4ffbf6SChris Packham 		mv_ddr_odpg_disable();
5612b4ffbf6SChris Packham 
5622b4ffbf6SChris Packham 		if (mv_ddr_is_odpg_done(MAX_POLLING_ITERATIONS) != MV_OK) {
5632b4ffbf6SChris Packham 			DEBUG_LEVELING(DEBUG_LEVEL_ERROR, ("odpg disable failed\n"));
564f1df9364SStefan Roese 			return MV_FAIL;
565f1df9364SStefan Roese 		}
566f1df9364SStefan Roese 
5672b4ffbf6SChris Packham 		ddr3_tip_if_write(0, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
5682b4ffbf6SChris Packham 				  ODPG_DATA_CTRL_REG, 0, MASK_ALL_BITS);
569f1df9364SStefan Roese 
570f1df9364SStefan Roese 		/* double loop on bus, pup */
571f1df9364SStefan Roese 		for (if_id = 0; if_id <= MAX_INTERFACE_NUM - 1; if_id++) {
5722b4ffbf6SChris Packham 			VALIDATE_IF_ACTIVE(tm->if_act_mask, if_id);
573f1df9364SStefan Roese 			/* check training done */
574f1df9364SStefan Roese 			for (bus_num = 0;
5752b4ffbf6SChris Packham 			     bus_num < octets_per_if_num;
576f1df9364SStefan Roese 			     bus_num++) {
5772b4ffbf6SChris Packham 				VALIDATE_BUS_ACTIVE(tm->bus_act_mask, bus_num);
578f1df9364SStefan Roese 
579f1df9364SStefan Roese 				if (per_bit_rl_pup_status[if_id][bus_num]
580f1df9364SStefan Roese 				    == 0) {
581f1df9364SStefan Roese 					curr_min_delay = 0;
582f1df9364SStefan Roese 					for (bit_num = 0; bit_num < 8;
583f1df9364SStefan Roese 					     bit_num++) {
584f1df9364SStefan Roese 						if (ddr3_tip_if_polling
585f1df9364SStefan Roese 						    (dev_num,
586f1df9364SStefan Roese 						     ACCESS_TYPE_UNICAST,
587f1df9364SStefan Roese 						     if_id, (1 << 25),
588f1df9364SStefan Roese 						     (1 << 25),
589f1df9364SStefan Roese 						     mask_results_dq_reg_map
590f1df9364SStefan Roese 						     [bus_num * 8 + bit_num],
591f1df9364SStefan Roese 						     MAX_POLLING_ITERATIONS) !=
592f1df9364SStefan Roese 						    MV_OK) {
593f1df9364SStefan Roese 							DEBUG_LEVELING
594f1df9364SStefan Roese 								(DEBUG_LEVEL_ERROR,
595f1df9364SStefan Roese 								 ("\n_r_l: DDR3 poll failed(2) for bus %d bit %d\n",
596f1df9364SStefan Roese 								  bus_num,
597f1df9364SStefan Roese 								  bit_num));
598f1df9364SStefan Roese 						} else {
599f1df9364SStefan Roese 							/* read result per pup */
600f1df9364SStefan Roese 							CHECK_STATUS
601f1df9364SStefan Roese 								(ddr3_tip_if_read
602f1df9364SStefan Roese 								 (dev_num,
603f1df9364SStefan Roese 								  ACCESS_TYPE_UNICAST,
604f1df9364SStefan Roese 								  if_id,
605f1df9364SStefan Roese 								  mask_results_dq_reg_map
606f1df9364SStefan Roese 								  [bus_num * 8 +
607f1df9364SStefan Roese 								   bit_num],
608f1df9364SStefan Roese 								  data_read,
609f1df9364SStefan Roese 								  MASK_ALL_BITS));
610f1df9364SStefan Roese 							data =
611f1df9364SStefan Roese 								(data_read
612f1df9364SStefan Roese 								 [if_id] &
613f1df9364SStefan Roese 								 0x1f) |
614f1df9364SStefan Roese 								((data_read
615f1df9364SStefan Roese 								  [if_id] &
616f1df9364SStefan Roese 								  0xe0) << 1);
617f1df9364SStefan Roese 							if (curr_min_delay == 0)
618f1df9364SStefan Roese 								curr_min_delay =
619f1df9364SStefan Roese 									data;
620f1df9364SStefan Roese 							else if (data <
621f1df9364SStefan Roese 								 curr_min_delay)
622f1df9364SStefan Roese 								curr_min_delay =
623f1df9364SStefan Roese 									data;
624f1df9364SStefan Roese 							if (data > data2_write[if_id][bus_num])
625f1df9364SStefan Roese 								data2_write
626f1df9364SStefan Roese 									[if_id]
627f1df9364SStefan Roese 									[bus_num] =
628f1df9364SStefan Roese 									data;
629f1df9364SStefan Roese 						}
630f1df9364SStefan Roese 					}
631f1df9364SStefan Roese 
632f1df9364SStefan Roese 					if (data2_write[if_id][bus_num] <=
633f1df9364SStefan Roese 					    (curr_min_delay +
634f1df9364SStefan Roese 					     MAX_DQ_READ_LEVELING_DELAY)) {
635f1df9364SStefan Roese 						per_bit_rl_pup_status[if_id]
636f1df9364SStefan Roese 							[bus_num] = 1;
637f1df9364SStefan Roese 					}
638f1df9364SStefan Roese 				}
639f1df9364SStefan Roese 			}
640f1df9364SStefan Roese 		}
641f1df9364SStefan Roese 
642f1df9364SStefan Roese 		/* check if there is need to search new phyreg3 value */
643f1df9364SStefan Roese 		if (curr_numb < 2) {
644f1df9364SStefan Roese 			/* if there is DLL that is not checked yet */
645f1df9364SStefan Roese 			for (if_id = 0; if_id <= MAX_INTERFACE_NUM - 1;
646f1df9364SStefan Roese 			     if_id++) {
6472b4ffbf6SChris Packham 				VALIDATE_IF_ACTIVE(tm->if_act_mask, if_id);
648f1df9364SStefan Roese 				for (bus_num = 0;
6492b4ffbf6SChris Packham 				     bus_num < octets_per_if_num;
650f1df9364SStefan Roese 				     bus_num++) {
6512b4ffbf6SChris Packham 					VALIDATE_BUS_ACTIVE(tm->bus_act_mask,
652f1df9364SStefan Roese 							bus_num);
653f1df9364SStefan Roese 					if (per_bit_rl_pup_status[if_id]
654f1df9364SStefan Roese 					    [bus_num] != 1) {
655f1df9364SStefan Roese 						/* go to next ADLL value */
656f1df9364SStefan Roese 						CHECK_STATUS
657f1df9364SStefan Roese 							(ddr3_tip_bus_write
658f1df9364SStefan Roese 							 (dev_num,
659f1df9364SStefan Roese 							  ACCESS_TYPE_UNICAST,
660f1df9364SStefan Roese 							  if_id,
661f1df9364SStefan Roese 							  ACCESS_TYPE_UNICAST,
662f1df9364SStefan Roese 							  bus_num, DDR_PHY_DATA,
6632b4ffbf6SChris Packham 							  CRX_PHY_REG(0),
664f1df9364SStefan Roese 							  (phyreg3_arr[if_id]
665f1df9364SStefan Roese 							   [bus_num] +
666f1df9364SStefan Roese 							   adll_array[curr_numb])));
667f1df9364SStefan Roese 						break_loop = 1;
668f1df9364SStefan Roese 						break;
669f1df9364SStefan Roese 					}
670f1df9364SStefan Roese 				}
671f1df9364SStefan Roese 				if (break_loop)
672f1df9364SStefan Roese 					break;
673f1df9364SStefan Roese 			}
674f1df9364SStefan Roese 		}		/* if (curr_numb < 2) */
675f1df9364SStefan Roese 		if (!break_loop)
676f1df9364SStefan Roese 			break;
677f1df9364SStefan Roese 	}		/* for ( curr_numb = 0; curr_numb <3; curr_numb++) */
678f1df9364SStefan Roese 
679f1df9364SStefan Roese 	for (if_id = 0; if_id <= MAX_INTERFACE_NUM - 1; if_id++) {
6802b4ffbf6SChris Packham 		VALIDATE_IF_ACTIVE(tm->if_act_mask, if_id);
6812b4ffbf6SChris Packham 		for (bus_num = 0; bus_num < octets_per_if_num;
682f1df9364SStefan Roese 		     bus_num++) {
6832b4ffbf6SChris Packham 			VALIDATE_BUS_ACTIVE(tm->bus_act_mask, bus_num);
684f1df9364SStefan Roese 			if (per_bit_rl_pup_status[if_id][bus_num] == 1)
685f1df9364SStefan Roese 				ddr3_tip_bus_write(dev_num,
686f1df9364SStefan Roese 						   ACCESS_TYPE_UNICAST,
687f1df9364SStefan Roese 						   if_id,
688f1df9364SStefan Roese 						   ACCESS_TYPE_UNICAST,
689f1df9364SStefan Roese 						   bus_num, DDR_PHY_DATA,
6902b4ffbf6SChris Packham 						   RL_PHY_REG(effective_cs),
691f1df9364SStefan Roese 						   data2_write[if_id]
692f1df9364SStefan Roese 						   [bus_num]);
693f1df9364SStefan Roese 			else
694f1df9364SStefan Roese 				is_any_pup_fail = 1;
695f1df9364SStefan Roese 		}
696f1df9364SStefan Roese 
697f1df9364SStefan Roese 		/* TBD flow does not support multi CS */
698f1df9364SStefan Roese 		/*
699f1df9364SStefan Roese 		 * cs_bitmask = tm->interface_params[if_id].
700f1df9364SStefan Roese 		 * as_bus_params[bus_num].cs_bitmask;
701f1df9364SStefan Roese 		 */
702f1df9364SStefan Roese 		/* divide by 4 is used for retrieving the CS number */
703f1df9364SStefan Roese 		/*
704f1df9364SStefan Roese 		 * TBD BC2 - what is the PHY address for other
705f1df9364SStefan Roese 		 * CS ddr3_tip_write_cs_result() ???
706f1df9364SStefan Roese 		 */
707f1df9364SStefan Roese 		/*
708f1df9364SStefan Roese 		 * find what should be written to PHY
709f1df9364SStefan Roese 		 * - max delay that is less than threshold
710f1df9364SStefan Roese 		 */
711f1df9364SStefan Roese 		if (is_any_pup_fail == 1) {
712f1df9364SStefan Roese 			training_result[training_stage][if_id] = TEST_FAILED;
713f1df9364SStefan Roese 			if (debug_mode == 0)
714f1df9364SStefan Roese 				return MV_FAIL;
715f1df9364SStefan Roese 		}
716f1df9364SStefan Roese 	}
717f1df9364SStefan Roese 	DEBUG_LEVELING(DEBUG_LEVEL_INFO, ("RL exit read leveling\n"));
718f1df9364SStefan Roese 
719f1df9364SStefan Roese 	/*
720f1df9364SStefan Roese 	 *     Phase 3: Exit Read Leveling
721f1df9364SStefan Roese 	 */
722f1df9364SStefan Roese 
723f1df9364SStefan Roese 	CHECK_STATUS(ddr3_tip_if_write
724f1df9364SStefan Roese 		     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
725f1df9364SStefan Roese 		      TRAINING_SW_2_REG, (1 << 3), (1 << 3)));
726f1df9364SStefan Roese 	CHECK_STATUS(ddr3_tip_if_write
727f1df9364SStefan Roese 		     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
728f1df9364SStefan Roese 		      TRAINING_SW_1_REG, (1 << 16), (1 << 16)));
729f1df9364SStefan Roese 	/* set ODPG to functional */
730f1df9364SStefan Roese 	CHECK_STATUS(ddr3_tip_if_write
731f1df9364SStefan Roese 		     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
7322b4ffbf6SChris Packham 		      ODPG_DATA_CTRL_REG, 0x0, MASK_ALL_BITS));
733f1df9364SStefan Roese 	/*
734f1df9364SStefan Roese 	 * Copy the result from the effective CS search to the real
735f1df9364SStefan Roese 	 * Functional CS
736f1df9364SStefan Roese 	 */
7372b4ffbf6SChris Packham 	ddr3_tip_write_cs_result(dev_num, RL_PHY_REG(0));
738f1df9364SStefan Roese 	CHECK_STATUS(ddr3_tip_if_write
739f1df9364SStefan Roese 		     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
7402b4ffbf6SChris Packham 		      ODPG_DATA_CTRL_REG, 0x0, MASK_ALL_BITS));
741f1df9364SStefan Roese 
742f1df9364SStefan Roese 	for (if_id = 0; if_id <= MAX_INTERFACE_NUM - 1; if_id++) {
7432b4ffbf6SChris Packham 		VALIDATE_IF_ACTIVE(tm->if_act_mask, if_id);
744f1df9364SStefan Roese 		/* restore cs enable value */
745f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
746f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_UNICAST, if_id,
7472b4ffbf6SChris Packham 			      DUAL_DUNIT_CFG_REG, cs_enable_reg_val[if_id],
748f1df9364SStefan Roese 			      MASK_ALL_BITS));
749f1df9364SStefan Roese 		if (odt_config != 0) {
750f1df9364SStefan Roese 			CHECK_STATUS(ddr3_tip_write_additional_odt_setting
751f1df9364SStefan Roese 				     (dev_num, if_id));
752f1df9364SStefan Roese 		}
753f1df9364SStefan Roese 	}
754f1df9364SStefan Roese 
755f1df9364SStefan Roese 	for (if_id = 0; if_id <= MAX_INTERFACE_NUM - 1; if_id++) {
7562b4ffbf6SChris Packham 		VALIDATE_IF_ACTIVE(tm->if_act_mask, if_id);
757f1df9364SStefan Roese 		if (training_result[training_stage][if_id] == TEST_FAILED)
758f1df9364SStefan Roese 			return MV_FAIL;
759f1df9364SStefan Roese 	}
760f1df9364SStefan Roese 
761f1df9364SStefan Roese 	return MV_OK;
762f1df9364SStefan Roese }
763f1df9364SStefan Roese 
ddr3_tip_calc_cs_mask(u32 dev_num,u32 if_id,u32 effective_cs,u32 * cs_mask)764f1df9364SStefan Roese int ddr3_tip_calc_cs_mask(u32 dev_num, u32 if_id, u32 effective_cs,
765f1df9364SStefan Roese 			  u32 *cs_mask)
766f1df9364SStefan Roese {
767f1df9364SStefan Roese 	u32 all_bus_cs = 0, same_bus_cs;
768f1df9364SStefan Roese 	u32 bus_cnt;
7692b4ffbf6SChris Packham 	u32 octets_per_if_num = ddr3_tip_dev_attr_get(dev_num, MV_ATTR_OCTET_PER_INTERFACE);
7702b4ffbf6SChris Packham 	struct mv_ddr_topology_map *tm = mv_ddr_topology_map_get();
771f1df9364SStefan Roese 
772f1df9364SStefan Roese 	*cs_mask = same_bus_cs = CS_BIT_MASK;
773f1df9364SStefan Roese 
774f1df9364SStefan Roese 	/*
775f1df9364SStefan Roese 	 * In some of the devices (such as BC2), the CS is per pup and there
776f1df9364SStefan Roese 	 * for mixed mode is valid on like other devices where CS configuration
777f1df9364SStefan Roese 	 * is per interface.
778f1df9364SStefan Roese 	 * In order to know that, we do 'Or' and 'And' operation between all
779f1df9364SStefan Roese 	 * CS (of the pups).
780f1df9364SStefan Roese 	 * If they are they are not the same then it's mixed mode so all CS
781f1df9364SStefan Roese 	 * should be configured (when configuring the MRS)
782f1df9364SStefan Roese 	 */
7832b4ffbf6SChris Packham 	for (bus_cnt = 0; bus_cnt < octets_per_if_num; bus_cnt++) {
7842b4ffbf6SChris Packham 		VALIDATE_BUS_ACTIVE(tm->bus_act_mask, bus_cnt);
785f1df9364SStefan Roese 
786f1df9364SStefan Roese 		all_bus_cs |= tm->interface_params[if_id].
787f1df9364SStefan Roese 			as_bus_params[bus_cnt].cs_bitmask;
788f1df9364SStefan Roese 		same_bus_cs &= tm->interface_params[if_id].
789f1df9364SStefan Roese 			as_bus_params[bus_cnt].cs_bitmask;
790f1df9364SStefan Roese 
791f1df9364SStefan Roese 		/* cs enable is active low */
792f1df9364SStefan Roese 		*cs_mask &= ~tm->interface_params[if_id].
793f1df9364SStefan Roese 			as_bus_params[bus_cnt].cs_bitmask;
794f1df9364SStefan Roese 	}
795f1df9364SStefan Roese 
796f1df9364SStefan Roese 	if (all_bus_cs == same_bus_cs)
797f1df9364SStefan Roese 		*cs_mask = (*cs_mask | (~(1 << effective_cs))) & CS_BIT_MASK;
798f1df9364SStefan Roese 
799f1df9364SStefan Roese 	return MV_OK;
800f1df9364SStefan Roese }
801f1df9364SStefan Roese 
802f1df9364SStefan Roese /*
803f1df9364SStefan Roese  * Dynamic write leveling
804f1df9364SStefan Roese  */
ddr3_tip_dynamic_write_leveling(u32 dev_num,int phase_remove)8052b4ffbf6SChris Packham int ddr3_tip_dynamic_write_leveling(u32 dev_num, int phase_remove)
806f1df9364SStefan Roese {
8072b4ffbf6SChris Packham 	u32 reg_data = 0, temp = 0, iter, if_id, bus_cnt;
808f1df9364SStefan Roese 	u32 cs_enable_reg_val[MAX_INTERFACE_NUM] = { 0 };
809f1df9364SStefan Roese 	u32 cs_mask[MAX_INTERFACE_NUM];
810f1df9364SStefan Roese 	u32 read_data_sample_delay_vals[MAX_INTERFACE_NUM] = { 0 };
811f1df9364SStefan Roese 	u32 read_data_ready_delay_vals[MAX_INTERFACE_NUM] = { 0 };
812f1df9364SStefan Roese 	/* 0 for failure */
813f1df9364SStefan Roese 	u32 res_values[MAX_INTERFACE_NUM * MAX_BUS_NUM] = { 0 };
814f1df9364SStefan Roese 	u32 test_res = 0;	/* 0 - success for all pup */
815f1df9364SStefan Roese 	u32 data_read[MAX_INTERFACE_NUM];
816*ebb1a593SChris Packham 	u8 wl_values[MAX_CS_NUM][MAX_BUS_NUM][MAX_INTERFACE_NUM];
817f1df9364SStefan Roese 	u16 *mask_results_pup_reg_map = ddr3_tip_get_mask_results_pup_reg_map();
818f1df9364SStefan Roese 	u32 cs_mask0[MAX_INTERFACE_NUM] = { 0 };
819*ebb1a593SChris Packham 	unsigned int max_cs = mv_ddr_cs_num_get();
8202b4ffbf6SChris Packham 	u32 octets_per_if_num = ddr3_tip_dev_attr_get(dev_num, MV_ATTR_OCTET_PER_INTERFACE);
8212b4ffbf6SChris Packham 	struct mv_ddr_topology_map *tm = mv_ddr_topology_map_get();
822f1df9364SStefan Roese 
823f1df9364SStefan Roese 	for (if_id = 0; if_id <= MAX_INTERFACE_NUM - 1; if_id++) {
8242b4ffbf6SChris Packham 		VALIDATE_IF_ACTIVE(tm->if_act_mask, if_id);
825f1df9364SStefan Roese 
826f1df9364SStefan Roese 		training_result[training_stage][if_id] = TEST_SUCCESS;
827f1df9364SStefan Roese 
828f1df9364SStefan Roese 		/* save Read Data Sample Delay */
829f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_read
830f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_UNICAST, if_id,
8312b4ffbf6SChris Packham 			      RD_DATA_SMPL_DLYS_REG,
832f1df9364SStefan Roese 			      read_data_sample_delay_vals, MASK_ALL_BITS));
833f1df9364SStefan Roese 		/* save Read Data Ready Delay */
834f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_read
835f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_UNICAST, if_id,
8362b4ffbf6SChris Packham 			      RD_DATA_RDY_DLYS_REG, read_data_ready_delay_vals,
837f1df9364SStefan Roese 			      MASK_ALL_BITS));
838f1df9364SStefan Roese 		/* save current cs reg val */
839f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_read
840f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_UNICAST, if_id,
8412b4ffbf6SChris Packham 			      DUAL_DUNIT_CFG_REG, cs_enable_reg_val, MASK_ALL_BITS));
8422b4ffbf6SChris Packham 	}
8432b4ffbf6SChris Packham 
8442b4ffbf6SChris Packham 	if (ddr3_tip_dev_attr_get(dev_num, MV_ATTR_TIP_REV) < MV_TIP_REV_3) {
8452b4ffbf6SChris Packham 		/* Enable multi-CS */
8462b4ffbf6SChris Packham 		CHECK_STATUS(ddr3_tip_if_write
8472b4ffbf6SChris Packham 			     (dev_num, ACCESS_TYPE_UNICAST, if_id,
8482b4ffbf6SChris Packham 			     DUAL_DUNIT_CFG_REG, 0, (1 << 3)));
849f1df9364SStefan Roese 	}
850f1df9364SStefan Roese 
851f1df9364SStefan Roese 	/*
852f1df9364SStefan Roese 	 *     Phase 1: DRAM 2 Write Leveling mode
853f1df9364SStefan Roese 	 */
854f1df9364SStefan Roese 
855f1df9364SStefan Roese 	/*Assert 10 refresh commands to DRAM to all CS */
856f1df9364SStefan Roese 	for (iter = 0; iter < WL_ITERATION_NUM; iter++) {
857f1df9364SStefan Roese 		for (if_id = 0; if_id <= MAX_INTERFACE_NUM - 1; if_id++) {
8582b4ffbf6SChris Packham 			VALIDATE_IF_ACTIVE(tm->if_act_mask, if_id);
859f1df9364SStefan Roese 			CHECK_STATUS(ddr3_tip_if_write
860f1df9364SStefan Roese 				     (dev_num, ACCESS_TYPE_UNICAST,
8612b4ffbf6SChris Packham 				      if_id, SDRAM_OP_REG,
862f1df9364SStefan Roese 				      (u32)((~(0xf) << 8) | 0x2), 0xf1f));
863f1df9364SStefan Roese 		}
864f1df9364SStefan Roese 	}
865f1df9364SStefan Roese 	/* check controller back to normal */
866f1df9364SStefan Roese 	for (if_id = 0; if_id <= MAX_INTERFACE_NUM - 1; if_id++) {
8672b4ffbf6SChris Packham 		VALIDATE_IF_ACTIVE(tm->if_act_mask, if_id);
868f1df9364SStefan Roese 		if (ddr3_tip_if_polling
869f1df9364SStefan Roese 		    (dev_num, ACCESS_TYPE_UNICAST, if_id, 0, 0x1f,
8702b4ffbf6SChris Packham 		     SDRAM_OP_REG, MAX_POLLING_ITERATIONS) != MV_OK) {
871f1df9364SStefan Roese 			DEBUG_LEVELING(DEBUG_LEVEL_ERROR,
872f1df9364SStefan Roese 				       ("WL: DDR3 poll failed(3)"));
873f1df9364SStefan Roese 		}
874f1df9364SStefan Roese 	}
875f1df9364SStefan Roese 
876f1df9364SStefan Roese 	for (effective_cs = 0; effective_cs < max_cs; effective_cs++) {
877f1df9364SStefan Roese 		/*enable write leveling to all cs  - Q off , WL n */
878f1df9364SStefan Roese 		/* calculate interface cs mask */
8792b4ffbf6SChris Packham 		CHECK_STATUS(ddr3_tip_write_mrs_cmd(dev_num, cs_mask0, MR_CMD1,
880f1df9364SStefan Roese 						    0x1000, 0x1080));
881f1df9364SStefan Roese 
882f1df9364SStefan Roese 		for (if_id = 0; if_id <= MAX_INTERFACE_NUM - 1; if_id++) {
8832b4ffbf6SChris Packham 			VALIDATE_IF_ACTIVE(tm->if_act_mask, if_id);
884f1df9364SStefan Roese 			/* cs enable is active low */
885f1df9364SStefan Roese 			ddr3_tip_calc_cs_mask(dev_num, if_id, effective_cs,
886f1df9364SStefan Roese 					      &cs_mask[if_id]);
887f1df9364SStefan Roese 		}
888f1df9364SStefan Roese 
8892b4ffbf6SChris Packham 		if (ddr3_tip_dev_attr_get(dev_num, MV_ATTR_TIP_REV) >= MV_TIP_REV_3) {
890f1df9364SStefan Roese 			/* Enable Output buffer to relevant CS - Q on , WL on */
891f1df9364SStefan Roese 			CHECK_STATUS(ddr3_tip_write_mrs_cmd
8922b4ffbf6SChris Packham 				     (dev_num, cs_mask, MR_CMD1, 0x80, 0x1080));
893f1df9364SStefan Roese 
894f1df9364SStefan Roese 			/*enable odt for relevant CS */
895f1df9364SStefan Roese 			CHECK_STATUS(ddr3_tip_if_write
896f1df9364SStefan Roese 				     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
897f1df9364SStefan Roese 				      0x1498, (0x3 << (effective_cs * 2)), 0xf));
8982b4ffbf6SChris Packham 		} else {
8992b4ffbf6SChris Packham 			/* FIXME: should be the same as _CPU case */
9002b4ffbf6SChris Packham 			CHECK_STATUS(ddr3_tip_write_mrs_cmd
9012b4ffbf6SChris Packham 				     (dev_num, cs_mask, MR_CMD1, 0xc0, 0x12c4));
9022b4ffbf6SChris Packham 		}
903f1df9364SStefan Roese 
904f1df9364SStefan Roese 		/*
905f1df9364SStefan Roese 		 *     Phase 2: Set training IP to write leveling mode
906f1df9364SStefan Roese 		 */
907f1df9364SStefan Roese 
908f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_dynamic_write_leveling_seq(dev_num));
909f1df9364SStefan Roese 
9102b4ffbf6SChris Packham 		/* phase 3: trigger training */
9112b4ffbf6SChris Packham 		mv_ddr_training_enable();
912f1df9364SStefan Roese 
9132b4ffbf6SChris Packham 		/* check for training done */
9142b4ffbf6SChris Packham 		if (mv_ddr_is_training_done(MAX_POLLING_ITERATIONS, data_read) != MV_OK) {
9152b4ffbf6SChris Packham 			DEBUG_LEVELING(DEBUG_LEVEL_ERROR, ("training done failed\n"));
9162b4ffbf6SChris Packham 		} else { /* check for training pass */
9172b4ffbf6SChris Packham 			reg_data = data_read[0];
9182b4ffbf6SChris Packham #if defined(CONFIG_ARMADA_38X) /* JIRA #1498 for 16 bit with ECC */
9192b4ffbf6SChris Packham 			if (tm->bus_act_mask == 0xb) /* set to data to 0 to skip the check */
9202b4ffbf6SChris Packham 				reg_data = 0;
921f1df9364SStefan Roese #endif
9222b4ffbf6SChris Packham 			if (reg_data != PASS)
9232b4ffbf6SChris Packham 				DEBUG_LEVELING(DEBUG_LEVEL_INFO, ("training result failed\n"));
924f1df9364SStefan Roese 
925f1df9364SStefan Roese 			/* check for training completion per bus */
9262b4ffbf6SChris Packham 			for (bus_cnt = 0; bus_cnt < octets_per_if_num; bus_cnt++) {
9272b4ffbf6SChris Packham 				VALIDATE_BUS_ACTIVE(tm->bus_act_mask, bus_cnt);
928f1df9364SStefan Roese 				/* training status */
9292b4ffbf6SChris Packham 				ddr3_tip_if_read(0, ACCESS_TYPE_UNICAST, 0,
9302b4ffbf6SChris Packham 					      mask_results_pup_reg_map[bus_cnt],
9312b4ffbf6SChris Packham 					      data_read, MASK_ALL_BITS);
9322b4ffbf6SChris Packham 				reg_data = data_read[0];
9332b4ffbf6SChris Packham 				DEBUG_LEVELING(DEBUG_LEVEL_TRACE, ("WL: IF %d BUS %d reg 0x%x\n",
9342b4ffbf6SChris Packham 								   0, bus_cnt, reg_data));
9352b4ffbf6SChris Packham 				if ((reg_data & (1 << 25)) == 0)
9362b4ffbf6SChris Packham 					res_values[bus_cnt] = 1;
9372b4ffbf6SChris Packham 				ddr3_tip_if_read(0, ACCESS_TYPE_UNICAST, 0,
9382b4ffbf6SChris Packham 					      mask_results_pup_reg_map[bus_cnt],
9392b4ffbf6SChris Packham 					      data_read, 0xff);
940f1df9364SStefan Roese 				/*
941f1df9364SStefan Roese 				 * Save the read value that should be
942f1df9364SStefan Roese 				 * write to PHY register
943f1df9364SStefan Roese 				 */
9442b4ffbf6SChris Packham 				wl_values[effective_cs][bus_cnt][0] = (u8)data_read[0];
9452b4ffbf6SChris Packham 			}
9462b4ffbf6SChris Packham 		}
9472b4ffbf6SChris Packham 
9482b4ffbf6SChris Packham 		/*
9492b4ffbf6SChris Packham 		 *     Phase 3.5: Validate result
9502b4ffbf6SChris Packham 		 */
9512b4ffbf6SChris Packham 		for (if_id = 0; if_id < MAX_INTERFACE_NUM; if_id++) {
9522b4ffbf6SChris Packham 			VALIDATE_IF_ACTIVE(tm->if_act_mask, if_id);
9532b4ffbf6SChris Packham 			for (bus_cnt = 0; bus_cnt < octets_per_if_num; bus_cnt++) {
9542b4ffbf6SChris Packham 				VALIDATE_BUS_ACTIVE(tm->bus_act_mask, bus_cnt);
9552b4ffbf6SChris Packham 				/*
9562b4ffbf6SChris Packham 				 * Read result control register according to subphy
9572b4ffbf6SChris Packham 				 * "16" below is for a half-phase
9582b4ffbf6SChris Packham 				 */
9592b4ffbf6SChris Packham 				reg_data = wl_values[effective_cs][bus_cnt][if_id] + 16;
9602b4ffbf6SChris Packham 				/*
9612b4ffbf6SChris Packham 				 * Write to WL register: ADLL [4:0], Phase [8:6],
9622b4ffbf6SChris Packham 				 * Centralization ADLL [15:10] + 0x10
9632b4ffbf6SChris Packham 				 */
9642b4ffbf6SChris Packham 				reg_data = (reg_data & 0x1f) |
9652b4ffbf6SChris Packham 					   (((reg_data & 0xe0) >> 5) << 6) |
9662b4ffbf6SChris Packham 					   (((reg_data & 0x1f) + phy_reg1_val) << 10);
9672b4ffbf6SChris Packham 				/* Search with WL CS0 subphy reg */
9682b4ffbf6SChris Packham 				ddr3_tip_bus_write(dev_num, ACCESS_TYPE_UNICAST, if_id,
9692b4ffbf6SChris Packham 						   ACCESS_TYPE_UNICAST, bus_cnt,
9702b4ffbf6SChris Packham 						   DDR_PHY_DATA, WL_PHY_REG(0), reg_data);
9712b4ffbf6SChris Packham 				/*
9722b4ffbf6SChris Packham 				 * Check for change in data read from DRAM.
9732b4ffbf6SChris Packham 				 * If changed, fix the result
9742b4ffbf6SChris Packham 				 */
9752b4ffbf6SChris Packham 				CHECK_STATUS(ddr3_tip_if_read
9762b4ffbf6SChris Packham 					     (dev_num,
9772b4ffbf6SChris Packham 					      ACCESS_TYPE_UNICAST,
9782b4ffbf6SChris Packham 					      if_id,
9792b4ffbf6SChris Packham 					      TRAINING_WL_REG,
9802b4ffbf6SChris Packham 					      data_read, MASK_ALL_BITS));
9812b4ffbf6SChris Packham 				if (((data_read[if_id] & (1 << (bus_cnt + 20))) >>
9822b4ffbf6SChris Packham 				     (bus_cnt + 20)) == 0) {
9832b4ffbf6SChris Packham 					DEBUG_LEVELING(
9842b4ffbf6SChris Packham 						DEBUG_LEVEL_ERROR,
9852b4ffbf6SChris Packham 						("WLValues was changed from 0x%X",
986f1df9364SStefan Roese 						 wl_values[effective_cs]
9872b4ffbf6SChris Packham 						 [bus_cnt][if_id]));
9882b4ffbf6SChris Packham 					wl_values[effective_cs]
9892b4ffbf6SChris Packham 					[bus_cnt][if_id] += 32;
9902b4ffbf6SChris Packham 					DEBUG_LEVELING(
9912b4ffbf6SChris Packham 						DEBUG_LEVEL_ERROR,
9922b4ffbf6SChris Packham 						("to 0x%X",
9932b4ffbf6SChris Packham 						 wl_values[effective_cs]
9942b4ffbf6SChris Packham 						 [bus_cnt][if_id]));
995f1df9364SStefan Roese 				}
996f1df9364SStefan Roese 			}
997f1df9364SStefan Roese 		}
998f1df9364SStefan Roese 
999f1df9364SStefan Roese 		/*
1000f1df9364SStefan Roese 		 *     Phase 4: Exit write leveling mode
1001f1df9364SStefan Roese 		 */
1002f1df9364SStefan Roese 
1003f1df9364SStefan Roese 		/* disable DQs toggling */
1004f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
1005f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
10062b4ffbf6SChris Packham 			      WL_DQS_PATTERN_REG, 0x0, 0x1));
1007f1df9364SStefan Roese 
1008f1df9364SStefan Roese 		/* Update MRS 1 (WL off) */
10092b4ffbf6SChris Packham 		if (ddr3_tip_dev_attr_get(dev_num, MV_ATTR_TIP_REV) >= MV_TIP_REV_3) {
10102b4ffbf6SChris Packham 			CHECK_STATUS(ddr3_tip_write_mrs_cmd(dev_num, cs_mask0, MR_CMD1,
1011f1df9364SStefan Roese 							    0x1000, 0x1080));
10122b4ffbf6SChris Packham 		} else {
10132b4ffbf6SChris Packham 			/* FIXME: should be same as _CPU case */
10142b4ffbf6SChris Packham 			CHECK_STATUS(ddr3_tip_write_mrs_cmd(dev_num, cs_mask0, MR_CMD1,
10152b4ffbf6SChris Packham 							    0x1000, 0x12c4));
10162b4ffbf6SChris Packham 		}
1017f1df9364SStefan Roese 
1018f1df9364SStefan Roese 		/* Update MRS 1 (return to functional mode - Q on , WL off) */
1019f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_write_mrs_cmd
10202b4ffbf6SChris Packham 			     (dev_num, cs_mask0, MR_CMD1, 0x0, 0x1080));
1021f1df9364SStefan Roese 
1022f1df9364SStefan Roese 		/* set phy to normal mode */
1023f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
1024f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
1025f1df9364SStefan Roese 			      TRAINING_SW_2_REG, 0x5, 0x7));
1026f1df9364SStefan Roese 
1027f1df9364SStefan Roese 		/* exit sw override mode  */
1028f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
1029f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
1030f1df9364SStefan Roese 			      TRAINING_SW_2_REG, 0x4, 0x7));
1031f1df9364SStefan Roese 	}
1032f1df9364SStefan Roese 
1033f1df9364SStefan Roese 	/*
1034f1df9364SStefan Roese 	 *     Phase 5: Load WL values to each PHY
1035f1df9364SStefan Roese 	 */
1036f1df9364SStefan Roese 
1037f1df9364SStefan Roese 	for (effective_cs = 0; effective_cs < max_cs; effective_cs++) {
1038f1df9364SStefan Roese 		for (if_id = 0; if_id <= MAX_INTERFACE_NUM - 1; if_id++) {
10392b4ffbf6SChris Packham 			VALIDATE_IF_ACTIVE(tm->if_act_mask, if_id);
1040f1df9364SStefan Roese 			test_res = 0;
1041f1df9364SStefan Roese 			for (bus_cnt = 0;
10422b4ffbf6SChris Packham 			     bus_cnt < octets_per_if_num;
1043f1df9364SStefan Roese 			     bus_cnt++) {
10442b4ffbf6SChris Packham 				VALIDATE_BUS_ACTIVE(tm->bus_act_mask, bus_cnt);
1045f1df9364SStefan Roese 				/* check if result == pass */
1046f1df9364SStefan Roese 				if (res_values
1047f1df9364SStefan Roese 				    [(if_id *
10482b4ffbf6SChris Packham 				      octets_per_if_num) +
1049f1df9364SStefan Roese 				     bus_cnt] == 0) {
1050f1df9364SStefan Roese 					/*
1051f1df9364SStefan Roese 					 * read result control register
1052f1df9364SStefan Roese 					 * according to pup
1053f1df9364SStefan Roese 					 */
1054f1df9364SStefan Roese 					reg_data =
1055f1df9364SStefan Roese 						wl_values[effective_cs][bus_cnt]
1056f1df9364SStefan Roese 						[if_id];
1057f1df9364SStefan Roese 					/*
1058f1df9364SStefan Roese 					 * Write into write leveling register
1059f1df9364SStefan Roese 					 * ([4:0] ADLL, [8:6] Phase, [15:10]
1060f1df9364SStefan Roese 					 * (centralization) ADLL + 0x10)
1061f1df9364SStefan Roese 					 */
1062f1df9364SStefan Roese 					reg_data =
1063f1df9364SStefan Roese 						(reg_data & 0x1f) |
1064f1df9364SStefan Roese 						(((reg_data & 0xe0) >> 5) << 6) |
1065f1df9364SStefan Roese 						(((reg_data & 0x1f) +
1066f1df9364SStefan Roese 						  phy_reg1_val) << 10);
10672b4ffbf6SChris Packham 					/*
10682b4ffbf6SChris Packham 					 * in case phase remove should be executed
10692b4ffbf6SChris Packham 					 * need to remove more than one phase.
10702b4ffbf6SChris Packham 					 * this will take place only in low frequency,
10712b4ffbf6SChris Packham 					 * where there could be more than one phase between sub-phys
10722b4ffbf6SChris Packham 					 */
10732b4ffbf6SChris Packham 					if (phase_remove == 1) {
10742b4ffbf6SChris Packham 						temp = (reg_data >> WR_LVL_PH_SEL_OFFS) & WR_LVL_PH_SEL_PHASE1;
10752b4ffbf6SChris Packham 						reg_data &= ~(WR_LVL_PH_SEL_MASK << WR_LVL_PH_SEL_OFFS);
10762b4ffbf6SChris Packham 						reg_data |= (temp << WR_LVL_PH_SEL_OFFS);
10772b4ffbf6SChris Packham 					}
10782b4ffbf6SChris Packham 
1079f1df9364SStefan Roese 					ddr3_tip_bus_write(
1080f1df9364SStefan Roese 						dev_num,
1081f1df9364SStefan Roese 						ACCESS_TYPE_UNICAST,
1082f1df9364SStefan Roese 						if_id,
1083f1df9364SStefan Roese 						ACCESS_TYPE_UNICAST,
1084f1df9364SStefan Roese 						bus_cnt,
1085f1df9364SStefan Roese 						DDR_PHY_DATA,
10862b4ffbf6SChris Packham 						WL_PHY_REG(effective_cs),
1087f1df9364SStefan Roese 						reg_data);
1088f1df9364SStefan Roese 				} else {
1089f1df9364SStefan Roese 					test_res = 1;
1090f1df9364SStefan Roese 					/*
1091f1df9364SStefan Roese 					 * read result control register
1092f1df9364SStefan Roese 					 * according to pup
1093f1df9364SStefan Roese 					 */
1094f1df9364SStefan Roese 					CHECK_STATUS(ddr3_tip_if_read
1095f1df9364SStefan Roese 						     (dev_num,
1096f1df9364SStefan Roese 						      ACCESS_TYPE_UNICAST,
1097f1df9364SStefan Roese 						      if_id,
1098f1df9364SStefan Roese 						      mask_results_pup_reg_map
1099f1df9364SStefan Roese 						      [bus_cnt], data_read,
1100f1df9364SStefan Roese 						      0xff));
1101f1df9364SStefan Roese 					reg_data = data_read[if_id];
1102f1df9364SStefan Roese 					DEBUG_LEVELING(
1103f1df9364SStefan Roese 						DEBUG_LEVEL_ERROR,
1104f1df9364SStefan Roese 						("WL: IF %d BUS %d failed, reg 0x%x\n",
1105f1df9364SStefan Roese 						 if_id, bus_cnt, reg_data));
1106f1df9364SStefan Roese 				}
1107f1df9364SStefan Roese 			}
1108f1df9364SStefan Roese 
1109f1df9364SStefan Roese 			if (test_res != 0) {
1110f1df9364SStefan Roese 				training_result[training_stage][if_id] =
1111f1df9364SStefan Roese 					TEST_FAILED;
1112f1df9364SStefan Roese 			}
1113f1df9364SStefan Roese 		}
1114f1df9364SStefan Roese 	}
1115f1df9364SStefan Roese 	/* Set to 0 after each loop to avoid illegal value may be used */
1116f1df9364SStefan Roese 	effective_cs = 0;
1117f1df9364SStefan Roese 
1118f1df9364SStefan Roese 	/*
1119f1df9364SStefan Roese 	 * Copy the result from the effective CS search to the real
1120f1df9364SStefan Roese 	 * Functional CS
1121f1df9364SStefan Roese 	 */
11222b4ffbf6SChris Packham 	/* ddr3_tip_write_cs_result(dev_num, WL_PHY_REG(0); */
1123f1df9364SStefan Roese 	/* restore saved values */
1124f1df9364SStefan Roese 	for (if_id = 0; if_id <= MAX_INTERFACE_NUM - 1; if_id++) {
11252b4ffbf6SChris Packham 		VALIDATE_IF_ACTIVE(tm->if_act_mask, if_id);
1126f1df9364SStefan Roese 		/* restore Read Data Sample Delay */
1127f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
1128f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_UNICAST, if_id,
11292b4ffbf6SChris Packham 			      RD_DATA_SMPL_DLYS_REG,
1130f1df9364SStefan Roese 			      read_data_sample_delay_vals[if_id],
1131f1df9364SStefan Roese 			      MASK_ALL_BITS));
1132f1df9364SStefan Roese 
1133f1df9364SStefan Roese 		/* restore Read Data Ready Delay */
1134f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
1135f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_UNICAST, if_id,
11362b4ffbf6SChris Packham 			      RD_DATA_RDY_DLYS_REG,
1137f1df9364SStefan Roese 			      read_data_ready_delay_vals[if_id],
1138f1df9364SStefan Roese 			      MASK_ALL_BITS));
1139f1df9364SStefan Roese 
1140f1df9364SStefan Roese 		/* enable multi cs */
1141f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
1142f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_UNICAST, if_id,
11432b4ffbf6SChris Packham 			      DUAL_DUNIT_CFG_REG, cs_enable_reg_val[if_id],
1144f1df9364SStefan Roese 			      MASK_ALL_BITS));
1145f1df9364SStefan Roese 	}
1146f1df9364SStefan Roese 
11472b4ffbf6SChris Packham 	if (ddr3_tip_dev_attr_get(dev_num, MV_ATTR_TIP_REV) >= MV_TIP_REV_3) {
11482b4ffbf6SChris Packham 		/* Disable modt0 for CS0 training - need to adjust for multi-CS
11492b4ffbf6SChris Packham 		 * in case of ddr4 set 0xf else 0
11502b4ffbf6SChris Packham 		 */
11512b4ffbf6SChris Packham 		if (odt_config != 0) {
11522b4ffbf6SChris Packham 			CHECK_STATUS(ddr3_tip_if_write(dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
11532b4ffbf6SChris Packham 						       SDRAM_ODT_CTRL_HIGH_REG, 0x0, 0xf));
11542b4ffbf6SChris Packham 		}
11552b4ffbf6SChris Packham 		else {
11562b4ffbf6SChris Packham 			CHECK_STATUS(ddr3_tip_if_write(dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
11572b4ffbf6SChris Packham 						       SDRAM_ODT_CTRL_HIGH_REG, 0xf, 0xf));
11582b4ffbf6SChris Packham 		}
11592b4ffbf6SChris Packham 
11602b4ffbf6SChris Packham 	}
1161f1df9364SStefan Roese 
1162f1df9364SStefan Roese 	for (if_id = 0; if_id <= MAX_INTERFACE_NUM - 1; if_id++) {
11632b4ffbf6SChris Packham 		VALIDATE_IF_ACTIVE(tm->if_act_mask, if_id);
1164f1df9364SStefan Roese 		if (training_result[training_stage][if_id] == TEST_FAILED)
1165f1df9364SStefan Roese 			return MV_FAIL;
1166f1df9364SStefan Roese 	}
1167f1df9364SStefan Roese 
1168f1df9364SStefan Roese 	return MV_OK;
1169f1df9364SStefan Roese }
1170f1df9364SStefan Roese 
1171f1df9364SStefan Roese /*
1172f1df9364SStefan Roese  * Dynamic write leveling supplementary
1173f1df9364SStefan Roese  */
ddr3_tip_dynamic_write_leveling_supp(u32 dev_num)1174f1df9364SStefan Roese int ddr3_tip_dynamic_write_leveling_supp(u32 dev_num)
1175f1df9364SStefan Roese {
1176f1df9364SStefan Roese 	int adll_offset;
1177f1df9364SStefan Roese 	u32 if_id, bus_id, data, data_tmp;
1178f1df9364SStefan Roese 	int is_if_fail = 0;
11792b4ffbf6SChris Packham 	u32 octets_per_if_num = ddr3_tip_dev_attr_get(dev_num, MV_ATTR_OCTET_PER_INTERFACE);
11802b4ffbf6SChris Packham 	struct mv_ddr_topology_map *tm = mv_ddr_topology_map_get();
1181f1df9364SStefan Roese 
1182f1df9364SStefan Roese 	for (if_id = 0; if_id <= MAX_INTERFACE_NUM - 1; if_id++) {
11832b4ffbf6SChris Packham 		VALIDATE_IF_ACTIVE(tm->if_act_mask, if_id);
1184f1df9364SStefan Roese 		is_if_fail = 0;
1185f1df9364SStefan Roese 
11862b4ffbf6SChris Packham 		for (bus_id = 0; bus_id < octets_per_if_num; bus_id++) {
11872b4ffbf6SChris Packham 			VALIDATE_BUS_ACTIVE(tm->bus_act_mask, bus_id);
1188f1df9364SStefan Roese 			wr_supp_res[if_id][bus_id].is_pup_fail = 1;
1189f1df9364SStefan Roese 			CHECK_STATUS(ddr3_tip_bus_read
1190f1df9364SStefan Roese 				     (dev_num, if_id, ACCESS_TYPE_UNICAST,
1191f1df9364SStefan Roese 				      bus_id, DDR_PHY_DATA,
11922b4ffbf6SChris Packham 				      CTX_PHY_REG(effective_cs),
1193f1df9364SStefan Roese 				      &data));
1194f1df9364SStefan Roese 			DEBUG_LEVELING(
1195f1df9364SStefan Roese 				DEBUG_LEVEL_TRACE,
1196f1df9364SStefan Roese 				("WL Supp: adll_offset=0 data delay = %d\n",
1197f1df9364SStefan Roese 				 data));
1198f1df9364SStefan Roese 			if (ddr3_tip_wl_supp_align_phase_shift
11992b4ffbf6SChris Packham 			    (dev_num, if_id, bus_id) == MV_OK) {
1200f1df9364SStefan Roese 				DEBUG_LEVELING(
1201f1df9364SStefan Roese 					DEBUG_LEVEL_TRACE,
1202f1df9364SStefan Roese 					("WL Supp: IF %d bus_id %d adll_offset=0 Success !\n",
1203f1df9364SStefan Roese 					 if_id, bus_id));
1204f1df9364SStefan Roese 				continue;
1205f1df9364SStefan Roese 			}
1206f1df9364SStefan Roese 
1207f1df9364SStefan Roese 			/* change adll */
1208f1df9364SStefan Roese 			adll_offset = 5;
1209f1df9364SStefan Roese 			CHECK_STATUS(ddr3_tip_bus_write
1210f1df9364SStefan Roese 				     (dev_num, ACCESS_TYPE_UNICAST, if_id,
1211f1df9364SStefan Roese 				      ACCESS_TYPE_UNICAST, bus_id, DDR_PHY_DATA,
12122b4ffbf6SChris Packham 				      CTX_PHY_REG(effective_cs),
1213f1df9364SStefan Roese 				      data + adll_offset));
1214f1df9364SStefan Roese 			CHECK_STATUS(ddr3_tip_bus_read
1215f1df9364SStefan Roese 				     (dev_num, if_id, ACCESS_TYPE_UNICAST,
1216f1df9364SStefan Roese 				      bus_id, DDR_PHY_DATA,
12172b4ffbf6SChris Packham 				      CTX_PHY_REG(effective_cs),
1218f1df9364SStefan Roese 				      &data_tmp));
1219f1df9364SStefan Roese 			DEBUG_LEVELING(
1220f1df9364SStefan Roese 				DEBUG_LEVEL_TRACE,
1221f1df9364SStefan Roese 				("WL Supp: adll_offset= %d data delay = %d\n",
1222f1df9364SStefan Roese 				 adll_offset, data_tmp));
1223f1df9364SStefan Roese 
1224f1df9364SStefan Roese 			if (ddr3_tip_wl_supp_align_phase_shift
12252b4ffbf6SChris Packham 			    (dev_num, if_id, bus_id) == MV_OK) {
1226f1df9364SStefan Roese 				DEBUG_LEVELING(
1227f1df9364SStefan Roese 					DEBUG_LEVEL_TRACE,
1228f1df9364SStefan Roese 					("WL Supp: IF %d bus_id %d adll_offset= %d Success !\n",
1229f1df9364SStefan Roese 					 if_id, bus_id, adll_offset));
1230f1df9364SStefan Roese 				continue;
1231f1df9364SStefan Roese 			}
1232f1df9364SStefan Roese 
1233f1df9364SStefan Roese 			/* change adll */
1234f1df9364SStefan Roese 			adll_offset = -5;
1235f1df9364SStefan Roese 			CHECK_STATUS(ddr3_tip_bus_write
1236f1df9364SStefan Roese 				     (dev_num, ACCESS_TYPE_UNICAST, if_id,
1237f1df9364SStefan Roese 				      ACCESS_TYPE_UNICAST, bus_id, DDR_PHY_DATA,
12382b4ffbf6SChris Packham 				      CTX_PHY_REG(effective_cs),
1239f1df9364SStefan Roese 				      data + adll_offset));
1240f1df9364SStefan Roese 			CHECK_STATUS(ddr3_tip_bus_read
1241f1df9364SStefan Roese 				     (dev_num, if_id, ACCESS_TYPE_UNICAST,
1242f1df9364SStefan Roese 				      bus_id, DDR_PHY_DATA,
12432b4ffbf6SChris Packham 				      CTX_PHY_REG(effective_cs),
1244f1df9364SStefan Roese 				      &data_tmp));
1245f1df9364SStefan Roese 			DEBUG_LEVELING(
1246f1df9364SStefan Roese 				DEBUG_LEVEL_TRACE,
1247f1df9364SStefan Roese 				("WL Supp: adll_offset= %d data delay = %d\n",
1248f1df9364SStefan Roese 				 adll_offset, data_tmp));
1249f1df9364SStefan Roese 			if (ddr3_tip_wl_supp_align_phase_shift
12502b4ffbf6SChris Packham 			    (dev_num, if_id, bus_id) == MV_OK) {
1251f1df9364SStefan Roese 				DEBUG_LEVELING(
1252f1df9364SStefan Roese 					DEBUG_LEVEL_TRACE,
1253f1df9364SStefan Roese 					("WL Supp: IF %d bus_id %d adll_offset= %d Success !\n",
1254f1df9364SStefan Roese 					 if_id, bus_id, adll_offset));
1255f1df9364SStefan Roese 				continue;
1256f1df9364SStefan Roese 			} else {
1257f1df9364SStefan Roese 				DEBUG_LEVELING(
1258f1df9364SStefan Roese 					DEBUG_LEVEL_ERROR,
1259f1df9364SStefan Roese 					("WL Supp: IF %d bus_id %d Failed !\n",
1260f1df9364SStefan Roese 					 if_id, bus_id));
1261f1df9364SStefan Roese 				is_if_fail = 1;
1262f1df9364SStefan Roese 			}
1263f1df9364SStefan Roese 		}
1264f1df9364SStefan Roese 
1265f1df9364SStefan Roese 		if (is_if_fail == 1) {
1266f1df9364SStefan Roese 			DEBUG_LEVELING(DEBUG_LEVEL_ERROR,
12672b4ffbf6SChris Packham 				       ("WL Supp: CS# %d: IF %d failed\n",
12682b4ffbf6SChris Packham 					effective_cs, if_id));
1269f1df9364SStefan Roese 			training_result[training_stage][if_id] = TEST_FAILED;
1270f1df9364SStefan Roese 		} else {
1271f1df9364SStefan Roese 			training_result[training_stage][if_id] = TEST_SUCCESS;
1272f1df9364SStefan Roese 		}
1273f1df9364SStefan Roese 	}
1274f1df9364SStefan Roese 
1275f1df9364SStefan Roese 	for (if_id = 0; if_id <= MAX_INTERFACE_NUM - 1; if_id++) {
12762b4ffbf6SChris Packham 		VALIDATE_IF_ACTIVE(tm->if_act_mask, if_id);
1277f1df9364SStefan Roese 		if (training_result[training_stage][if_id] == TEST_FAILED)
1278f1df9364SStefan Roese 			return MV_FAIL;
1279f1df9364SStefan Roese 	}
1280f1df9364SStefan Roese 
1281f1df9364SStefan Roese 	return MV_OK;
1282f1df9364SStefan Roese }
1283f1df9364SStefan Roese 
1284f1df9364SStefan Roese /*
1285f1df9364SStefan Roese  * Phase Shift
1286f1df9364SStefan Roese  */
ddr3_tip_wl_supp_align_phase_shift(u32 dev_num,u32 if_id,u32 bus_id)1287f1df9364SStefan Roese static int ddr3_tip_wl_supp_align_phase_shift(u32 dev_num, u32 if_id,
12882b4ffbf6SChris Packham 					      u32 bus_id)
1289f1df9364SStefan Roese {
12902b4ffbf6SChris Packham 	u32 original_phase;
12912b4ffbf6SChris Packham 	u32 data, write_data;
12922b4ffbf6SChris Packham 
1293f1df9364SStefan Roese 	wr_supp_res[if_id][bus_id].stage = PHASE_SHIFT;
12942b4ffbf6SChris Packham 	if (ddr3_tip_xsb_compare_test
12952b4ffbf6SChris Packham 	    (dev_num, if_id, bus_id, 0) == MV_OK)
1296f1df9364SStefan Roese 		return MV_OK;
12972b4ffbf6SChris Packham 
12982b4ffbf6SChris Packham 	/* Read current phase */
12992b4ffbf6SChris Packham 	CHECK_STATUS(ddr3_tip_bus_read
13002b4ffbf6SChris Packham 		     (dev_num, if_id, ACCESS_TYPE_UNICAST, bus_id,
13012b4ffbf6SChris Packham 		      DDR_PHY_DATA, WL_PHY_REG(effective_cs), &data));
13022b4ffbf6SChris Packham 	original_phase = (data >> 6) & 0x7;
13032b4ffbf6SChris Packham 
13042b4ffbf6SChris Packham 	/* Set phase (0x0[6-8]) -2 */
13052b4ffbf6SChris Packham 	if (original_phase >= 1) {
13062b4ffbf6SChris Packham 		if (original_phase == 1)
13072b4ffbf6SChris Packham 			write_data = data & ~0x1df;
13082b4ffbf6SChris Packham 		else
13092b4ffbf6SChris Packham 			write_data = (data & ~0x1c0) |
13102b4ffbf6SChris Packham 				     ((original_phase - 2) << 6);
13112b4ffbf6SChris Packham 		ddr3_tip_bus_write(dev_num, ACCESS_TYPE_UNICAST, if_id,
13122b4ffbf6SChris Packham 				   ACCESS_TYPE_UNICAST, bus_id, DDR_PHY_DATA,
13132b4ffbf6SChris Packham 				   WL_PHY_REG(effective_cs), write_data);
13142b4ffbf6SChris Packham 		if (ddr3_tip_xsb_compare_test
13152b4ffbf6SChris Packham 		    (dev_num, if_id, bus_id, -2) == MV_OK)
1316f1df9364SStefan Roese 			return MV_OK;
1317f1df9364SStefan Roese 	}
13182b4ffbf6SChris Packham 
13192b4ffbf6SChris Packham 	/* Set phase (0x0[6-8]) +2 */
13202b4ffbf6SChris Packham 	if (original_phase <= 5) {
13212b4ffbf6SChris Packham 		write_data = (data & ~0x1c0) |
13222b4ffbf6SChris Packham 			     ((original_phase + 2) << 6);
13232b4ffbf6SChris Packham 		ddr3_tip_bus_write(dev_num, ACCESS_TYPE_UNICAST, if_id,
13242b4ffbf6SChris Packham 				   ACCESS_TYPE_UNICAST, bus_id, DDR_PHY_DATA,
13252b4ffbf6SChris Packham 				   WL_PHY_REG(effective_cs), write_data);
13262b4ffbf6SChris Packham 		if (ddr3_tip_xsb_compare_test
13272b4ffbf6SChris Packham 		    (dev_num, if_id, bus_id, 2) == MV_OK)
13282b4ffbf6SChris Packham 			return MV_OK;
13292b4ffbf6SChris Packham 	}
13302b4ffbf6SChris Packham 
13312b4ffbf6SChris Packham 	/* Set phase (0x0[6-8]) +4 */
13322b4ffbf6SChris Packham 	if (original_phase <= 3) {
13332b4ffbf6SChris Packham 		write_data = (data & ~0x1c0) |
13342b4ffbf6SChris Packham 			     ((original_phase + 4) << 6);
13352b4ffbf6SChris Packham 		ddr3_tip_bus_write(dev_num, ACCESS_TYPE_UNICAST, if_id,
13362b4ffbf6SChris Packham 				   ACCESS_TYPE_UNICAST, bus_id, DDR_PHY_DATA,
13372b4ffbf6SChris Packham 				   WL_PHY_REG(effective_cs), write_data);
13382b4ffbf6SChris Packham 		if (ddr3_tip_xsb_compare_test
13392b4ffbf6SChris Packham 		    (dev_num, if_id, bus_id, 4) == MV_OK)
13402b4ffbf6SChris Packham 			return MV_OK;
13412b4ffbf6SChris Packham 	}
13422b4ffbf6SChris Packham 
13432b4ffbf6SChris Packham 	/* Set phase (0x0[6-8]) +6 */
13442b4ffbf6SChris Packham 	if (original_phase <= 1) {
13452b4ffbf6SChris Packham 		write_data = (data & ~0x1c0) |
13462b4ffbf6SChris Packham 			     ((original_phase + 6) << 6);
13472b4ffbf6SChris Packham 		ddr3_tip_bus_write(dev_num, ACCESS_TYPE_UNICAST, if_id,
13482b4ffbf6SChris Packham 				   ACCESS_TYPE_UNICAST, bus_id, DDR_PHY_DATA,
13492b4ffbf6SChris Packham 				   WL_PHY_REG(effective_cs), write_data);
13502b4ffbf6SChris Packham 		if (ddr3_tip_xsb_compare_test
13512b4ffbf6SChris Packham 		    (dev_num, if_id, bus_id, 6) == MV_OK)
13522b4ffbf6SChris Packham 			return MV_OK;
13532b4ffbf6SChris Packham 	}
13542b4ffbf6SChris Packham 
13552b4ffbf6SChris Packham 	/* Write original WL result back */
13562b4ffbf6SChris Packham 	ddr3_tip_bus_write(dev_num, ACCESS_TYPE_UNICAST, if_id,
13572b4ffbf6SChris Packham 			   ACCESS_TYPE_UNICAST, bus_id, DDR_PHY_DATA,
13582b4ffbf6SChris Packham 			   WL_PHY_REG(effective_cs), data);
13592b4ffbf6SChris Packham 	wr_supp_res[if_id][bus_id].is_pup_fail = 1;
13602b4ffbf6SChris Packham 
13612b4ffbf6SChris Packham 	return MV_FAIL;
1362f1df9364SStefan Roese }
1363f1df9364SStefan Roese 
1364f1df9364SStefan Roese /*
1365f1df9364SStefan Roese  * Compare Test
1366f1df9364SStefan Roese  */
ddr3_tip_xsb_compare_test(u32 dev_num,u32 if_id,u32 bus_id,u32 edge_offset)1367f1df9364SStefan Roese static int ddr3_tip_xsb_compare_test(u32 dev_num, u32 if_id, u32 bus_id,
13682b4ffbf6SChris Packham 				     u32 edge_offset)
1369f1df9364SStefan Roese {
13702b4ffbf6SChris Packham 	u32 num_of_succ_byte_compare, word_in_pattern;
13712b4ffbf6SChris Packham 	u32 word_offset, i, num_of_word_mult;
1372f1df9364SStefan Roese 	u32 read_pattern[TEST_PATTERN_LENGTH * 2];
1373f1df9364SStefan Roese 	struct pattern_info *pattern_table = ddr3_tip_get_pattern_table();
1374f1df9364SStefan Roese 	u32 pattern_test_pattern_table[8];
13752b4ffbf6SChris Packham 	struct mv_ddr_topology_map *tm = mv_ddr_topology_map_get();
13762b4ffbf6SChris Packham 
13772b4ffbf6SChris Packham 	/* 3 below for INTERFACE_BUS_MASK_16BIT */
13782b4ffbf6SChris Packham 	num_of_word_mult = (tm->bus_act_mask == 3) ? 1 : 2;
1379f1df9364SStefan Roese 
1380f1df9364SStefan Roese 	for (i = 0; i < 8; i++) {
1381f1df9364SStefan Roese 		pattern_test_pattern_table[i] =
1382f1df9364SStefan Roese 			pattern_table_get_word(dev_num, PATTERN_TEST, (u8)i);
1383f1df9364SStefan Roese 	}
1384f1df9364SStefan Roese 
13852b4ffbf6SChris Packham 	/* External write, read and compare */
13862b4ffbf6SChris Packham 	CHECK_STATUS(ddr3_tip_load_pattern_to_mem(dev_num, PATTERN_TEST));
1387f1df9364SStefan Roese 
1388f1df9364SStefan Roese 	CHECK_STATUS(ddr3_tip_reset_fifo_ptr(dev_num));
1389f1df9364SStefan Roese 
1390f1df9364SStefan Roese 	CHECK_STATUS(ddr3_tip_ext_read
1391f1df9364SStefan Roese 		     (dev_num, if_id,
13922b4ffbf6SChris Packham 		      ((pattern_table[PATTERN_TEST].start_addr << 3) +
1393f1df9364SStefan Roese 		       ((SDRAM_CS_SIZE + 1) * effective_cs)), 1, read_pattern));
1394f1df9364SStefan Roese 
1395f1df9364SStefan Roese 	DEBUG_LEVELING(
1396f1df9364SStefan Roese 		DEBUG_LEVEL_TRACE,
13972b4ffbf6SChris Packham 		("XSB-compt CS#%d: IF %d bus_id %d 0x%x 0x%x 0x%x 0x%x 0x%x 0x%x 0x%x 0x%x\n",
13982b4ffbf6SChris Packham 		 effective_cs, if_id, bus_id,
13992b4ffbf6SChris Packham 		 read_pattern[0], read_pattern[1],
14002b4ffbf6SChris Packham 		 read_pattern[2], read_pattern[3],
14012b4ffbf6SChris Packham 		 read_pattern[4], read_pattern[5],
14022b4ffbf6SChris Packham 		 read_pattern[6], read_pattern[7]));
1403f1df9364SStefan Roese 
1404f1df9364SStefan Roese 	/* compare byte per pup */
1405f1df9364SStefan Roese 	num_of_succ_byte_compare = 0;
1406f1df9364SStefan Roese 	for (word_in_pattern = start_xsb_offset;
14072b4ffbf6SChris Packham 	     word_in_pattern < (TEST_PATTERN_LENGTH * num_of_word_mult);
14082b4ffbf6SChris Packham 	     word_in_pattern++) {
14092b4ffbf6SChris Packham 		word_offset = word_in_pattern;
14102b4ffbf6SChris Packham 		if ((word_offset > (TEST_PATTERN_LENGTH * 2 - 1)))
1411f1df9364SStefan Roese 			continue;
1412f1df9364SStefan Roese 
1413f1df9364SStefan Roese 		if ((read_pattern[word_in_pattern] & pup_mask_table[bus_id]) ==
1414f1df9364SStefan Roese 		    (pattern_test_pattern_table[word_offset] &
1415f1df9364SStefan Roese 		     pup_mask_table[bus_id]))
1416f1df9364SStefan Roese 			num_of_succ_byte_compare++;
1417f1df9364SStefan Roese 	}
1418f1df9364SStefan Roese 
14192b4ffbf6SChris Packham 	if ((TEST_PATTERN_LENGTH * num_of_word_mult - start_xsb_offset) ==
14202b4ffbf6SChris Packham 	    num_of_succ_byte_compare) {
14212b4ffbf6SChris Packham 		wr_supp_res[if_id][bus_id].stage = edge_offset;
14222b4ffbf6SChris Packham 		DEBUG_LEVELING(DEBUG_LEVEL_TRACE,
14232b4ffbf6SChris Packham 			       ("supplementary: shift to %d for if %d pup %d success\n",
14242b4ffbf6SChris Packham 				edge_offset, if_id, bus_id));
14252b4ffbf6SChris Packham 		wr_supp_res[if_id][bus_id].is_pup_fail = 0;
14262b4ffbf6SChris Packham 
1427f1df9364SStefan Roese 		return MV_OK;
1428f1df9364SStefan Roese 	} else {
1429f1df9364SStefan Roese 		DEBUG_LEVELING(
1430f1df9364SStefan Roese 			DEBUG_LEVEL_TRACE,
14312b4ffbf6SChris Packham 			("XSB-compt CS#%d: IF %d bus_id %d num_of_succ_byte_compare %d - Fail!\n",
14322b4ffbf6SChris Packham 			 effective_cs, if_id, bus_id, num_of_succ_byte_compare));
1433f1df9364SStefan Roese 
1434f1df9364SStefan Roese 		DEBUG_LEVELING(
1435f1df9364SStefan Roese 			DEBUG_LEVEL_TRACE,
1436f1df9364SStefan Roese 			("XSB-compt: expected 0x%x 0x%x 0x%x 0x%x 0x%x 0x%x 0x%x 0x%x\n",
1437f1df9364SStefan Roese 			 pattern_test_pattern_table[0],
1438f1df9364SStefan Roese 			 pattern_test_pattern_table[1],
1439f1df9364SStefan Roese 			 pattern_test_pattern_table[2],
1440f1df9364SStefan Roese 			 pattern_test_pattern_table[3],
1441f1df9364SStefan Roese 			 pattern_test_pattern_table[4],
1442f1df9364SStefan Roese 			 pattern_test_pattern_table[5],
1443f1df9364SStefan Roese 			 pattern_test_pattern_table[6],
1444f1df9364SStefan Roese 			 pattern_test_pattern_table[7]));
1445f1df9364SStefan Roese 		DEBUG_LEVELING(
1446f1df9364SStefan Roese 			DEBUG_LEVEL_TRACE,
1447f1df9364SStefan Roese 			("XSB-compt: recieved 0x%x 0x%x 0x%x 0x%x 0x%x 0x%x 0x%x 0x%x\n",
1448f1df9364SStefan Roese 			 read_pattern[0], read_pattern[1],
1449f1df9364SStefan Roese 			 read_pattern[2], read_pattern[3],
1450f1df9364SStefan Roese 			 read_pattern[4], read_pattern[5],
1451f1df9364SStefan Roese 			 read_pattern[6], read_pattern[7]));
1452f1df9364SStefan Roese 
1453f1df9364SStefan Roese 		return MV_FAIL;
1454f1df9364SStefan Roese 	}
1455f1df9364SStefan Roese }
1456f1df9364SStefan Roese 
1457f1df9364SStefan Roese /*
1458f1df9364SStefan Roese  * Dynamic write leveling sequence
1459f1df9364SStefan Roese  */
ddr3_tip_dynamic_write_leveling_seq(u32 dev_num)1460f1df9364SStefan Roese static int ddr3_tip_dynamic_write_leveling_seq(u32 dev_num)
1461f1df9364SStefan Roese {
1462f1df9364SStefan Roese 	u32 bus_id, dq_id;
1463f1df9364SStefan Roese 	u16 *mask_results_pup_reg_map = ddr3_tip_get_mask_results_pup_reg_map();
1464f1df9364SStefan Roese 	u16 *mask_results_dq_reg_map = ddr3_tip_get_mask_results_dq_reg();
14652b4ffbf6SChris Packham 	u32 octets_per_if_num = ddr3_tip_dev_attr_get(dev_num, MV_ATTR_OCTET_PER_INTERFACE);
14662b4ffbf6SChris Packham 	struct mv_ddr_topology_map *tm = mv_ddr_topology_map_get();
1467f1df9364SStefan Roese 
1468f1df9364SStefan Roese 	CHECK_STATUS(ddr3_tip_if_write
1469f1df9364SStefan Roese 		     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
1470f1df9364SStefan Roese 		      TRAINING_SW_2_REG, 0x1, 0x5));
1471f1df9364SStefan Roese 	CHECK_STATUS(ddr3_tip_if_write
1472f1df9364SStefan Roese 		     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
14732b4ffbf6SChris Packham 		      TRAINING_WL_REG, 0x50, 0xff));
1474f1df9364SStefan Roese 	CHECK_STATUS(ddr3_tip_if_write
1475f1df9364SStefan Roese 		     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
14762b4ffbf6SChris Packham 		      TRAINING_WL_REG, 0x5c, 0xff));
1477f1df9364SStefan Roese 	CHECK_STATUS(ddr3_tip_if_write
1478f1df9364SStefan Roese 		     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
14792b4ffbf6SChris Packham 		      GENERAL_TRAINING_OPCODE_REG, 0x381b82, 0x3c3faf));
1480f1df9364SStefan Roese 	CHECK_STATUS(ddr3_tip_if_write
1481f1df9364SStefan Roese 		     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
14822b4ffbf6SChris Packham 		      OPCODE_REG0_REG(1), (0x3 << 25), (0x3ffff << 9)));
1483f1df9364SStefan Roese 	CHECK_STATUS(ddr3_tip_if_write
1484f1df9364SStefan Roese 		     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
14852b4ffbf6SChris Packham 		      OPCODE_REG1_REG(1), 0x80, 0xffff));
1486f1df9364SStefan Roese 	CHECK_STATUS(ddr3_tip_if_write
1487f1df9364SStefan Roese 		     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
14882b4ffbf6SChris Packham 		      WL_DONE_CNTR_REF_REG, 0x14, 0xff));
1489f1df9364SStefan Roese 	CHECK_STATUS(ddr3_tip_if_write
1490f1df9364SStefan Roese 		     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
14912b4ffbf6SChris Packham 		      TRAINING_WL_REG, 0xff5c, 0xffff));
1492f1df9364SStefan Roese 
1493f1df9364SStefan Roese 	/* mask PBS */
1494f1df9364SStefan Roese 	for (dq_id = 0; dq_id < MAX_DQ_NUM; dq_id++) {
1495f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
1496f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
1497f1df9364SStefan Roese 			      mask_results_dq_reg_map[dq_id], 0x1 << 24,
1498f1df9364SStefan Roese 			      0x1 << 24));
1499f1df9364SStefan Roese 	}
1500f1df9364SStefan Roese 
1501f1df9364SStefan Roese 	/* Mask all results */
15022b4ffbf6SChris Packham 	for (bus_id = 0; bus_id < octets_per_if_num; bus_id++) {
1503f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
1504f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
1505f1df9364SStefan Roese 			      mask_results_pup_reg_map[bus_id], 0x1 << 24,
1506f1df9364SStefan Roese 			      0x1 << 24));
1507f1df9364SStefan Roese 	}
1508f1df9364SStefan Roese 
1509f1df9364SStefan Roese 	/* Unmask only wanted */
15102b4ffbf6SChris Packham 	for (bus_id = 0; bus_id < octets_per_if_num; bus_id++) {
15112b4ffbf6SChris Packham 		VALIDATE_BUS_ACTIVE(tm->bus_act_mask, bus_id);
1512f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
1513f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
1514f1df9364SStefan Roese 			      mask_results_pup_reg_map[bus_id], 0, 0x1 << 24));
1515f1df9364SStefan Roese 	}
1516f1df9364SStefan Roese 
1517f1df9364SStefan Roese 	CHECK_STATUS(ddr3_tip_if_write
1518f1df9364SStefan Roese 		     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
15192b4ffbf6SChris Packham 		      WL_DQS_PATTERN_REG, 0x1, 0x1));
1520f1df9364SStefan Roese 
1521f1df9364SStefan Roese 	return MV_OK;
1522f1df9364SStefan Roese }
1523f1df9364SStefan Roese 
1524f1df9364SStefan Roese /*
1525f1df9364SStefan Roese  * Dynamic read leveling sequence
1526f1df9364SStefan Roese  */
ddr3_tip_dynamic_read_leveling_seq(u32 dev_num)1527f1df9364SStefan Roese static int ddr3_tip_dynamic_read_leveling_seq(u32 dev_num)
1528f1df9364SStefan Roese {
1529f1df9364SStefan Roese 	u32 bus_id, dq_id;
1530f1df9364SStefan Roese 	u16 *mask_results_pup_reg_map = ddr3_tip_get_mask_results_pup_reg_map();
1531f1df9364SStefan Roese 	u16 *mask_results_dq_reg_map = ddr3_tip_get_mask_results_dq_reg();
15322b4ffbf6SChris Packham 	u32 octets_per_if_num = ddr3_tip_dev_attr_get(dev_num, MV_ATTR_OCTET_PER_INTERFACE);
15332b4ffbf6SChris Packham 	struct mv_ddr_topology_map *tm = mv_ddr_topology_map_get();
1534f1df9364SStefan Roese 
1535f1df9364SStefan Roese 	/* mask PBS */
1536f1df9364SStefan Roese 	for (dq_id = 0; dq_id < MAX_DQ_NUM; dq_id++) {
1537f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
1538f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
1539f1df9364SStefan Roese 			      mask_results_dq_reg_map[dq_id], 0x1 << 24,
1540f1df9364SStefan Roese 			      0x1 << 24));
1541f1df9364SStefan Roese 	}
1542f1df9364SStefan Roese 
1543f1df9364SStefan Roese 	/* Mask all results */
15442b4ffbf6SChris Packham 	for (bus_id = 0; bus_id < octets_per_if_num; bus_id++) {
1545f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
1546f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
1547f1df9364SStefan Roese 			      mask_results_pup_reg_map[bus_id], 0x1 << 24,
1548f1df9364SStefan Roese 			      0x1 << 24));
1549f1df9364SStefan Roese 	}
1550f1df9364SStefan Roese 
1551f1df9364SStefan Roese 	/* Unmask only wanted */
15522b4ffbf6SChris Packham 	for (bus_id = 0; bus_id < octets_per_if_num; bus_id++) {
15532b4ffbf6SChris Packham 		VALIDATE_BUS_ACTIVE(tm->bus_act_mask, bus_id);
1554f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
1555f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
1556f1df9364SStefan Roese 			      mask_results_pup_reg_map[bus_id], 0, 0x1 << 24));
1557f1df9364SStefan Roese 	}
1558f1df9364SStefan Roese 
1559f1df9364SStefan Roese 	return MV_OK;
1560f1df9364SStefan Roese }
1561f1df9364SStefan Roese 
1562f1df9364SStefan Roese /*
1563f1df9364SStefan Roese  * Dynamic read leveling sequence
1564f1df9364SStefan Roese  */
ddr3_tip_dynamic_per_bit_read_leveling_seq(u32 dev_num)1565f1df9364SStefan Roese static int ddr3_tip_dynamic_per_bit_read_leveling_seq(u32 dev_num)
1566f1df9364SStefan Roese {
1567f1df9364SStefan Roese 	u32 bus_id, dq_id;
1568f1df9364SStefan Roese 	u16 *mask_results_pup_reg_map = ddr3_tip_get_mask_results_pup_reg_map();
1569f1df9364SStefan Roese 	u16 *mask_results_dq_reg_map = ddr3_tip_get_mask_results_dq_reg();
15702b4ffbf6SChris Packham 	u32 octets_per_if_num = ddr3_tip_dev_attr_get(dev_num, MV_ATTR_OCTET_PER_INTERFACE);
15712b4ffbf6SChris Packham 	struct mv_ddr_topology_map *tm = mv_ddr_topology_map_get();
1572f1df9364SStefan Roese 
1573f1df9364SStefan Roese 	/* mask PBS */
1574f1df9364SStefan Roese 	for (dq_id = 0; dq_id < MAX_DQ_NUM; dq_id++) {
1575f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
1576f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
1577f1df9364SStefan Roese 			      mask_results_dq_reg_map[dq_id], 0x1 << 24,
1578f1df9364SStefan Roese 			      0x1 << 24));
1579f1df9364SStefan Roese 	}
1580f1df9364SStefan Roese 
1581f1df9364SStefan Roese 	/* Mask all results */
15822b4ffbf6SChris Packham 	for (bus_id = 0; bus_id < octets_per_if_num; bus_id++) {
1583f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
1584f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
1585f1df9364SStefan Roese 			      mask_results_pup_reg_map[bus_id], 0x1 << 24,
1586f1df9364SStefan Roese 			      0x1 << 24));
1587f1df9364SStefan Roese 	}
1588f1df9364SStefan Roese 
1589f1df9364SStefan Roese 	/* Unmask only wanted */
1590f1df9364SStefan Roese 	for (dq_id = 0; dq_id < MAX_DQ_NUM; dq_id++) {
15912b4ffbf6SChris Packham 		VALIDATE_BUS_ACTIVE(tm->bus_act_mask, dq_id / 8);
1592f1df9364SStefan Roese 		CHECK_STATUS(ddr3_tip_if_write
1593f1df9364SStefan Roese 			     (dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE,
1594f1df9364SStefan Roese 			      mask_results_dq_reg_map[dq_id], 0x0 << 24,
1595f1df9364SStefan Roese 			      0x1 << 24));
1596f1df9364SStefan Roese 	}
1597f1df9364SStefan Roese 
1598f1df9364SStefan Roese 	return MV_OK;
1599f1df9364SStefan Roese }
1600f1df9364SStefan Roese 
1601f1df9364SStefan Roese /*
1602f1df9364SStefan Roese  * Print write leveling supplementary results
1603f1df9364SStefan Roese  */
ddr3_tip_print_wl_supp_result(u32 dev_num)1604f1df9364SStefan Roese int ddr3_tip_print_wl_supp_result(u32 dev_num)
1605f1df9364SStefan Roese {
1606f1df9364SStefan Roese 	u32 bus_id = 0, if_id = 0;
16072b4ffbf6SChris Packham 	u32 octets_per_if_num = ddr3_tip_dev_attr_get(dev_num, MV_ATTR_OCTET_PER_INTERFACE);
16082b4ffbf6SChris Packham 	struct mv_ddr_topology_map *tm = mv_ddr_topology_map_get();
1609f1df9364SStefan Roese 
1610f1df9364SStefan Roese 	DEBUG_LEVELING(DEBUG_LEVEL_INFO,
1611f1df9364SStefan Roese 		       ("I/F0 PUP0 Result[0 - success, 1-fail] ...\n"));
1612f1df9364SStefan Roese 
1613f1df9364SStefan Roese 	for (if_id = 0; if_id <= MAX_INTERFACE_NUM - 1; if_id++) {
16142b4ffbf6SChris Packham 		VALIDATE_IF_ACTIVE(tm->if_act_mask, if_id);
16152b4ffbf6SChris Packham 		for (bus_id = 0; bus_id < octets_per_if_num;
1616f1df9364SStefan Roese 		     bus_id++) {
16172b4ffbf6SChris Packham 			VALIDATE_BUS_ACTIVE(tm->bus_act_mask, bus_id);
1618f1df9364SStefan Roese 			DEBUG_LEVELING(DEBUG_LEVEL_INFO,
1619f1df9364SStefan Roese 				       ("%d ,", wr_supp_res[if_id]
1620f1df9364SStefan Roese 					[bus_id].is_pup_fail));
1621f1df9364SStefan Roese 		}
1622f1df9364SStefan Roese 	}
1623f1df9364SStefan Roese 	DEBUG_LEVELING(
1624f1df9364SStefan Roese 		DEBUG_LEVEL_INFO,
1625f1df9364SStefan Roese 		("I/F0 PUP0 Stage[0-phase_shift, 1-clock_shift, 2-align_shift] ...\n"));
1626f1df9364SStefan Roese 
1627f1df9364SStefan Roese 	for (if_id = 0; if_id <= MAX_INTERFACE_NUM - 1; if_id++) {
16282b4ffbf6SChris Packham 		VALIDATE_IF_ACTIVE(tm->if_act_mask, if_id);
16292b4ffbf6SChris Packham 		for (bus_id = 0; bus_id < octets_per_if_num;
1630f1df9364SStefan Roese 		     bus_id++) {
16312b4ffbf6SChris Packham 			VALIDATE_BUS_ACTIVE(tm->bus_act_mask, bus_id);
1632f1df9364SStefan Roese 			DEBUG_LEVELING(DEBUG_LEVEL_INFO,
1633f1df9364SStefan Roese 				       ("%d ,", wr_supp_res[if_id]
1634f1df9364SStefan Roese 					[bus_id].stage));
1635f1df9364SStefan Roese 		}
1636f1df9364SStefan Roese 	}
1637f1df9364SStefan Roese 
1638f1df9364SStefan Roese 	return MV_OK;
1639f1df9364SStefan Roese }
16402b4ffbf6SChris Packham 
16412b4ffbf6SChris Packham #define RD_FIFO_PTR_LOW_STAT_INDIR_ADDR		0x9a
16422b4ffbf6SChris Packham #define RD_FIFO_PTR_HIGH_STAT_INDIR_ADDR	0x9b
16432b4ffbf6SChris Packham /* position of falling dqs edge in fifo; walking 1 */
16442b4ffbf6SChris Packham #define RD_FIFO_DQS_FALL_EDGE_POS_0		0x1
16452b4ffbf6SChris Packham #define RD_FIFO_DQS_FALL_EDGE_POS_1		0x2
16462b4ffbf6SChris Packham #define RD_FIFO_DQS_FALL_EDGE_POS_2		0x4
16472b4ffbf6SChris Packham #define RD_FIFO_DQS_FALL_EDGE_POS_3		0x8
16482b4ffbf6SChris Packham #define RD_FIFO_DQS_FALL_EDGE_POS_4		0x10 /* lock */
16492b4ffbf6SChris Packham /* position of rising dqs edge in fifo; walking 0 */
16502b4ffbf6SChris Packham #define RD_FIFO_DQS_RISE_EDGE_POS_0		0x1fff
16512b4ffbf6SChris Packham #define RD_FIFO_DQS_RISE_EDGE_POS_1		0x3ffe
16522b4ffbf6SChris Packham #define RD_FIFO_DQS_RISE_EDGE_POS_2		0x3ffd
16532b4ffbf6SChris Packham #define RD_FIFO_DQS_RISE_EDGE_POS_3		0x3ffb
16542b4ffbf6SChris Packham #define RD_FIFO_DQS_RISE_EDGE_POS_4		0x3ff7 /* lock */
16552b4ffbf6SChris Packham #define TEST_ADDR		0x8
16562b4ffbf6SChris Packham #define TAPS_PER_UI		32
16572b4ffbf6SChris Packham #define UI_PER_RD_SAMPLE	4
16582b4ffbf6SChris Packham #define TAPS_PER_RD_SAMPLE	((UI_PER_RD_SAMPLE) * (TAPS_PER_UI))
16592b4ffbf6SChris Packham #define MAX_RD_SAMPLES		32
16602b4ffbf6SChris Packham #define MAX_RL_VALUE		((MAX_RD_SAMPLES) * (TAPS_PER_RD_SAMPLE))
16612b4ffbf6SChris Packham #define RD_FIFO_DLY		8
16622b4ffbf6SChris Packham #define STEP_SIZE		64
16632b4ffbf6SChris Packham #define RL_JITTER_WIDTH_LMT	20
16642b4ffbf6SChris Packham #define ADLL_TAPS_IN_CYCLE	64
16652b4ffbf6SChris Packham 
16662b4ffbf6SChris Packham enum rl_dqs_burst_state {
16672b4ffbf6SChris Packham 	RL_AHEAD = 0,
16682b4ffbf6SChris Packham 	RL_INSIDE,
16692b4ffbf6SChris Packham 	RL_BEHIND
16702b4ffbf6SChris Packham };
1671*ebb1a593SChris Packham 
1672*ebb1a593SChris Packham 
mv_ddr_rl_dqs_burst(u32 dev_num,u32 if_id,u32 freq)16732b4ffbf6SChris Packham int mv_ddr_rl_dqs_burst(u32 dev_num, u32 if_id, u32 freq)
16742b4ffbf6SChris Packham {
1675*ebb1a593SChris Packham 	enum rl_dqs_burst_state rl_state[MAX_CS_NUM][MAX_BUS_NUM][MAX_INTERFACE_NUM] = { { {0} } };
16762b4ffbf6SChris Packham 	enum hws_ddr_phy subphy_type = DDR_PHY_DATA;
16772b4ffbf6SChris Packham 	struct mv_ddr_topology_map *tm = mv_ddr_topology_map_get();
16782b4ffbf6SChris Packham 	int cl_val = tm->interface_params[0].cas_l;
16792b4ffbf6SChris Packham 	int rl_adll_val, rl_phase_val, sdr_cycle_incr, rd_sample, rd_ready;
16802b4ffbf6SChris Packham 	int final_rd_sample, final_rd_ready;
16812b4ffbf6SChris Packham 	int i, subphy_id, step;
16822b4ffbf6SChris Packham 	int pass_lock_num = 0;
16832b4ffbf6SChris Packham 	int init_pass_lock_num;
16842b4ffbf6SChris Packham 	int phase_delta;
16852b4ffbf6SChris Packham 	int min_phase, max_phase;
1686*ebb1a593SChris Packham 	unsigned int max_cs = mv_ddr_cs_num_get();
1687*ebb1a593SChris Packham 	u32 rl_values[MAX_CS_NUM][MAX_BUS_NUM][MAX_INTERFACE_NUM] = { { {0} } };
1688*ebb1a593SChris Packham 	u32 rl_min_values[MAX_CS_NUM][MAX_BUS_NUM][MAX_INTERFACE_NUM] = { { {0} } };
1689*ebb1a593SChris Packham 	u32 rl_max_values[MAX_CS_NUM][MAX_BUS_NUM][MAX_INTERFACE_NUM] = { { {0} } };
1690*ebb1a593SChris Packham 	u32 rl_val, rl_min_val[MAX_CS_NUM], rl_max_val[MAX_CS_NUM];
16912b4ffbf6SChris Packham 	u32 reg_val_low, reg_val_high;
16922b4ffbf6SChris Packham 	u32 reg_val, reg_mask;
16932b4ffbf6SChris Packham 	uintptr_t test_addr = TEST_ADDR;
16942b4ffbf6SChris Packham 
1695*ebb1a593SChris Packham 
16962b4ffbf6SChris Packham 	/* initialization */
1697*ebb1a593SChris Packham 	if (mv_ddr_is_ecc_ena()) {
16982b4ffbf6SChris Packham 		ddr3_tip_if_read(dev_num, ACCESS_TYPE_UNICAST, if_id, TRAINING_SW_2_REG,
16992b4ffbf6SChris Packham 				 &reg_val, MASK_ALL_BITS);
17002b4ffbf6SChris Packham 		reg_mask = (TRAINING_ECC_MUX_MASK << TRAINING_ECC_MUX_OFFS) |
17012b4ffbf6SChris Packham 			   (TRAINING_SW_OVRD_MASK << TRAINING_SW_OVRD_OFFS);
17022b4ffbf6SChris Packham 		reg_val &= ~reg_mask;
17032b4ffbf6SChris Packham 		reg_val |= (TRAINING_ECC_MUX_DIS << TRAINING_ECC_MUX_OFFS) |
17042b4ffbf6SChris Packham 			   (TRAINING_SW_OVRD_ENA << TRAINING_SW_OVRD_OFFS);
17052b4ffbf6SChris Packham 		ddr3_tip_if_write(0, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE, TRAINING_SW_2_REG,
17062b4ffbf6SChris Packham 				  reg_val, MASK_ALL_BITS);
17072b4ffbf6SChris Packham 		ddr3_tip_if_read(dev_num, ACCESS_TYPE_UNICAST, if_id, TRAINING_REG,
17082b4ffbf6SChris Packham 				 &reg_val, MASK_ALL_BITS);
17092b4ffbf6SChris Packham 		reg_mask = (TRN_START_MASK << TRN_START_OFFS);
17102b4ffbf6SChris Packham 		reg_val &= ~reg_mask;
17112b4ffbf6SChris Packham 		reg_val |= TRN_START_ENA << TRN_START_OFFS;
17122b4ffbf6SChris Packham 		ddr3_tip_if_write(0, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE, TRAINING_REG,
17132b4ffbf6SChris Packham 				  reg_val, MASK_ALL_BITS);
17142b4ffbf6SChris Packham 	}
17152b4ffbf6SChris Packham 
17162b4ffbf6SChris Packham 	for (effective_cs = 0; effective_cs < max_cs; effective_cs++)
17172b4ffbf6SChris Packham 		for (subphy_id = 0; subphy_id < MAX_BUS_NUM; subphy_id++)
17182b4ffbf6SChris Packham 			for (if_id = 0; if_id < MAX_INTERFACE_NUM; if_id++)
17192b4ffbf6SChris Packham 				if (IS_BUS_ACTIVE(tm->bus_act_mask, subphy_id) == 0)
17202b4ffbf6SChris Packham 					pass_lock_num++; /* increment on inactive subphys */
17212b4ffbf6SChris Packham 
17222b4ffbf6SChris Packham 	init_pass_lock_num = pass_lock_num / max_cs;
17232b4ffbf6SChris Packham 	for (effective_cs = 0; effective_cs < max_cs; effective_cs++) {
17242b4ffbf6SChris Packham 		for (if_id = 0; if_id < MAX_INTERFACE_NUM; if_id++) {
17252b4ffbf6SChris Packham 			VALIDATE_IF_ACTIVE(tm->if_act_mask, if_id);
17262b4ffbf6SChris Packham 			training_result[training_stage][if_id] = TEST_SUCCESS;
17272b4ffbf6SChris Packham 		}
17282b4ffbf6SChris Packham 	}
17292b4ffbf6SChris Packham 
17302b4ffbf6SChris Packham 	/* search for dqs edges per subphy */
17312b4ffbf6SChris Packham 	if_id = 0;
17322b4ffbf6SChris Packham 	for (effective_cs = 0; effective_cs < max_cs; effective_cs++) {
1733*ebb1a593SChris Packham 
17342b4ffbf6SChris Packham 		pass_lock_num = init_pass_lock_num;
17352b4ffbf6SChris Packham 		ddr3_tip_if_write(0, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE, ODPG_DATA_CTRL_REG,
17362b4ffbf6SChris Packham 				  effective_cs << ODPG_DATA_CS_OFFS,
17372b4ffbf6SChris Packham 				  ODPG_DATA_CS_MASK << ODPG_DATA_CS_OFFS);
17382b4ffbf6SChris Packham 		rl_min_val[effective_cs] = MAX_RL_VALUE;
17392b4ffbf6SChris Packham 		rl_max_val[effective_cs] = 0;
17402b4ffbf6SChris Packham 		step = STEP_SIZE;
17412b4ffbf6SChris Packham 		for (i = 0; i < MAX_RL_VALUE; i += step) {
17422b4ffbf6SChris Packham 			rl_val = 0;
17432b4ffbf6SChris Packham 			sdr_cycle_incr = i / TAPS_PER_RD_SAMPLE; /* sdr cycle increment */
17442b4ffbf6SChris Packham 			rd_sample = cl_val + 2 * sdr_cycle_incr;
17452b4ffbf6SChris Packham 			/* fifo out to in delay in search is constant */
17462b4ffbf6SChris Packham 			rd_ready = rd_sample + RD_FIFO_DLY;
17472b4ffbf6SChris Packham 
17482b4ffbf6SChris Packham 			ddr3_tip_if_write(0, ACCESS_TYPE_UNICAST, 0, RD_DATA_SMPL_DLYS_REG,
17492b4ffbf6SChris Packham 					  rd_sample << RD_SMPL_DLY_CS_OFFS(effective_cs),
17502b4ffbf6SChris Packham 					  RD_SMPL_DLY_CS_MASK << RD_SMPL_DLY_CS_OFFS(effective_cs));
17512b4ffbf6SChris Packham 			ddr3_tip_if_write(0, ACCESS_TYPE_UNICAST, 0, RD_DATA_RDY_DLYS_REG,
17522b4ffbf6SChris Packham 					  rd_ready << RD_RDY_DLY_CS_OFFS(effective_cs),
17532b4ffbf6SChris Packham 					  RD_RDY_DLY_CS_MASK << RD_RDY_DLY_CS_OFFS(effective_cs));
17542b4ffbf6SChris Packham 
17552b4ffbf6SChris Packham 			/* one sdr (single data rate) cycle incremented on every four phases of ddr clock */
17562b4ffbf6SChris Packham 			sdr_cycle_incr = i % TAPS_PER_RD_SAMPLE;
17572b4ffbf6SChris Packham 			rl_adll_val = sdr_cycle_incr % MAX_RD_SAMPLES;
17582b4ffbf6SChris Packham 			rl_phase_val = sdr_cycle_incr / MAX_RD_SAMPLES;
17592b4ffbf6SChris Packham 			rl_val = ((rl_adll_val & RL_REF_DLY_MASK) << RL_REF_DLY_OFFS) |
17602b4ffbf6SChris Packham 				 ((rl_phase_val & RL_PH_SEL_MASK) << RL_PH_SEL_OFFS);
17612b4ffbf6SChris Packham 
17622b4ffbf6SChris Packham 			/* write to all subphys (even to not connected or locked) */
17632b4ffbf6SChris Packham 			ddr3_tip_bus_write(dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE, ACCESS_TYPE_MULTICAST,
17642b4ffbf6SChris Packham 					   0, DDR_PHY_DATA, RL_PHY_REG(effective_cs), rl_val);
17652b4ffbf6SChris Packham 
17662b4ffbf6SChris Packham 			/* reset read fifo assertion */
17672b4ffbf6SChris Packham 			ddr3_tip_if_write(dev_num, ACCESS_TYPE_MULTICAST, if_id, SDRAM_CFG_REG,
17682b4ffbf6SChris Packham 					  DATA_PUP_RD_RESET_ENA << DATA_PUP_RD_RESET_OFFS,
17692b4ffbf6SChris Packham 					  DATA_PUP_RD_RESET_MASK << DATA_PUP_RD_RESET_OFFS);
17702b4ffbf6SChris Packham 
17712b4ffbf6SChris Packham 			/* reset read fifo deassertion */
17722b4ffbf6SChris Packham 			ddr3_tip_if_write(dev_num, ACCESS_TYPE_MULTICAST, if_id, SDRAM_CFG_REG,
17732b4ffbf6SChris Packham 					  DATA_PUP_RD_RESET_DIS << DATA_PUP_RD_RESET_OFFS,
17742b4ffbf6SChris Packham 					  DATA_PUP_RD_RESET_MASK << DATA_PUP_RD_RESET_OFFS);
17752b4ffbf6SChris Packham 
17762b4ffbf6SChris Packham 			/* perform one read burst */
17772b4ffbf6SChris Packham 			if (MV_DDR_IS_64BIT_DRAM_MODE(tm->bus_act_mask))
17782b4ffbf6SChris Packham 				readq(test_addr);
17792b4ffbf6SChris Packham 			else
17802b4ffbf6SChris Packham 				readl(test_addr);
17812b4ffbf6SChris Packham 
17822b4ffbf6SChris Packham 			/* progress read ptr; decide on rl state per byte */
17832b4ffbf6SChris Packham 			for (subphy_id = 0; subphy_id < MAX_BUS_NUM; subphy_id++) {
17842b4ffbf6SChris Packham 				if (rl_state[effective_cs][subphy_id][if_id] == RL_BEHIND)
17852b4ffbf6SChris Packham 					continue; /* skip locked subphys */
17862b4ffbf6SChris Packham 				ddr3_tip_bus_read(dev_num, if_id, ACCESS_TYPE_UNICAST, subphy_id, DDR_PHY_DATA,
17872b4ffbf6SChris Packham 						  RD_FIFO_PTR_LOW_STAT_INDIR_ADDR, &reg_val_low);
17882b4ffbf6SChris Packham 				ddr3_tip_bus_read(dev_num, if_id, ACCESS_TYPE_UNICAST, subphy_id, DDR_PHY_DATA,
17892b4ffbf6SChris Packham 						  RD_FIFO_PTR_HIGH_STAT_INDIR_ADDR, &reg_val_high);
17902b4ffbf6SChris Packham 				DEBUG_LEVELING(DEBUG_LEVEL_TRACE,
17912b4ffbf6SChris Packham 					       ("%s: cs %d, step %d, subphy %d, state %d, low 0x%04x, high 0x%04x; move to ",
17922b4ffbf6SChris Packham 						__func__, effective_cs, i, subphy_id,
17932b4ffbf6SChris Packham 						rl_state[effective_cs][subphy_id][if_id],
17942b4ffbf6SChris Packham 						reg_val_low, reg_val_high));
17952b4ffbf6SChris Packham 
17962b4ffbf6SChris Packham 				switch (rl_state[effective_cs][subphy_id][if_id]) {
17972b4ffbf6SChris Packham 				case RL_AHEAD:
17982b4ffbf6SChris Packham 					/* improve search resolution getting closer to the window */
17992b4ffbf6SChris Packham 					if (reg_val_low == RD_FIFO_DQS_FALL_EDGE_POS_4 &&
18002b4ffbf6SChris Packham 					    reg_val_high == RD_FIFO_DQS_RISE_EDGE_POS_4) {
18012b4ffbf6SChris Packham 						rl_state[effective_cs][subphy_id][if_id] = RL_INSIDE;
18022b4ffbf6SChris Packham 						rl_values[effective_cs][subphy_id][if_id] = i;
18032b4ffbf6SChris Packham 						rl_min_values[effective_cs][subphy_id][if_id] = i;
18042b4ffbf6SChris Packham 						DEBUG_LEVELING(DEBUG_LEVEL_TRACE,
18052b4ffbf6SChris Packham 							       ("new state %d\n",
18062b4ffbf6SChris Packham 								rl_state[effective_cs][subphy_id][if_id]));
18072b4ffbf6SChris Packham 					} else if (reg_val_low == RD_FIFO_DQS_FALL_EDGE_POS_3 &&
18082b4ffbf6SChris Packham 						   reg_val_high == RD_FIFO_DQS_RISE_EDGE_POS_3) {
18092b4ffbf6SChris Packham 						step = (step < 2) ? step : 2;
18102b4ffbf6SChris Packham 					} else if (reg_val_low == RD_FIFO_DQS_FALL_EDGE_POS_2 &&
18112b4ffbf6SChris Packham 						   reg_val_high == RD_FIFO_DQS_RISE_EDGE_POS_2) {
18122b4ffbf6SChris Packham 						step = (step < 16) ? step : 16;
18132b4ffbf6SChris Packham 					} else if (reg_val_low == RD_FIFO_DQS_FALL_EDGE_POS_1 &&
18142b4ffbf6SChris Packham 						   reg_val_high == RD_FIFO_DQS_RISE_EDGE_POS_1) {
18152b4ffbf6SChris Packham 						step = (step < 32) ? step : 32;
18162b4ffbf6SChris Packham 					} else if (reg_val_low == RD_FIFO_DQS_FALL_EDGE_POS_0 &&
18172b4ffbf6SChris Packham 						   reg_val_high == RD_FIFO_DQS_RISE_EDGE_POS_0) {
18182b4ffbf6SChris Packham 						step = (step < 64) ? step : 64;
18192b4ffbf6SChris Packham 					} else {
18202b4ffbf6SChris Packham 						/* otherwise, step is unchanged */
18212b4ffbf6SChris Packham 					}
18222b4ffbf6SChris Packham 					break;
18232b4ffbf6SChris Packham 				case RL_INSIDE:
18242b4ffbf6SChris Packham 					if (reg_val_low == RD_FIFO_DQS_FALL_EDGE_POS_4 &&
18252b4ffbf6SChris Packham 					    reg_val_high == RD_FIFO_DQS_RISE_EDGE_POS_4) {
18262b4ffbf6SChris Packham 						rl_max_values[effective_cs][subphy_id][if_id] = i;
18272b4ffbf6SChris Packham 						if ((rl_max_values[effective_cs][subphy_id][if_id] -
18282b4ffbf6SChris Packham 						     rl_min_values[effective_cs][subphy_id][if_id]) >
18292b4ffbf6SChris Packham 						    ADLL_TAPS_IN_CYCLE) {
18302b4ffbf6SChris Packham 							rl_state[effective_cs][subphy_id][if_id] = RL_BEHIND;
18312b4ffbf6SChris Packham 							rl_values[effective_cs][subphy_id][if_id] =
18322b4ffbf6SChris Packham 								(i + rl_values[effective_cs][subphy_id][if_id]) / 2;
18332b4ffbf6SChris Packham 							pass_lock_num++;
18342b4ffbf6SChris Packham 							DEBUG_LEVELING(DEBUG_LEVEL_TRACE,
18352b4ffbf6SChris Packham 								       ("new lock %d\n", pass_lock_num));
18362b4ffbf6SChris Packham 							if (rl_min_val[effective_cs] >
18372b4ffbf6SChris Packham 							    rl_values[effective_cs][subphy_id][if_id])
18382b4ffbf6SChris Packham 								rl_min_val[effective_cs] =
18392b4ffbf6SChris Packham 									rl_values[effective_cs][subphy_id][if_id];
18402b4ffbf6SChris Packham 							if (rl_max_val[effective_cs] <
18412b4ffbf6SChris Packham 							    rl_values[effective_cs][subphy_id][if_id])
18422b4ffbf6SChris Packham 								rl_max_val[effective_cs] =
18432b4ffbf6SChris Packham 									rl_values[effective_cs][subphy_id][if_id];
18442b4ffbf6SChris Packham 							step = 2;
18452b4ffbf6SChris Packham 						}
18462b4ffbf6SChris Packham 					}
18472b4ffbf6SChris Packham 					if (reg_val_low != RD_FIFO_DQS_FALL_EDGE_POS_4 ||
18482b4ffbf6SChris Packham 					    reg_val_high != RD_FIFO_DQS_RISE_EDGE_POS_4) {
18492b4ffbf6SChris Packham 						if ((i - rl_values[effective_cs][subphy_id][if_id]) <
18502b4ffbf6SChris Packham 						    RL_JITTER_WIDTH_LMT) {
18512b4ffbf6SChris Packham 							/* inside the jitter; not valid segment */
18522b4ffbf6SChris Packham 							rl_state[effective_cs][subphy_id][if_id] = RL_AHEAD;
18532b4ffbf6SChris Packham 							DEBUG_LEVELING(DEBUG_LEVEL_TRACE,
18542b4ffbf6SChris Packham 								       ("new state %d; jitter on mask\n",
18552b4ffbf6SChris Packham 									rl_state[effective_cs][subphy_id][if_id]));
18562b4ffbf6SChris Packham 						} else { /* finished valid segment */
18572b4ffbf6SChris Packham 							rl_state[effective_cs][subphy_id][if_id] = RL_BEHIND;
18582b4ffbf6SChris Packham 							rl_values[effective_cs][subphy_id][if_id] =
18592b4ffbf6SChris Packham 								(i + rl_values[effective_cs][subphy_id][if_id]) / 2;
18602b4ffbf6SChris Packham 							DEBUG_LEVELING(DEBUG_LEVEL_TRACE,
18612b4ffbf6SChris Packham 								       ("new state %d, solution %d\n",
18622b4ffbf6SChris Packham 									rl_state[effective_cs][subphy_id][if_id],
18632b4ffbf6SChris Packham 									rl_values[effective_cs][subphy_id][if_id]));
18642b4ffbf6SChris Packham 							pass_lock_num++;
18652b4ffbf6SChris Packham 							DEBUG_LEVELING(DEBUG_LEVEL_TRACE,
18662b4ffbf6SChris Packham 								       ("new lock %d\n", pass_lock_num));
18672b4ffbf6SChris Packham 							if (rl_min_val[effective_cs] >
18682b4ffbf6SChris Packham 							    rl_values[effective_cs][subphy_id][if_id])
18692b4ffbf6SChris Packham 								rl_min_val[effective_cs] =
18702b4ffbf6SChris Packham 									rl_values[effective_cs][subphy_id][if_id];
18712b4ffbf6SChris Packham 							if (rl_max_val[effective_cs] <
18722b4ffbf6SChris Packham 							    rl_values[effective_cs][subphy_id][if_id])
18732b4ffbf6SChris Packham 								rl_max_val[effective_cs] =
18742b4ffbf6SChris Packham 									rl_values[effective_cs][subphy_id][if_id];
18752b4ffbf6SChris Packham 							step = 2;
18762b4ffbf6SChris Packham 						}
18772b4ffbf6SChris Packham 					}
18782b4ffbf6SChris Packham 					break;
18792b4ffbf6SChris Packham 				case RL_BEHIND: /* do nothing */
18802b4ffbf6SChris Packham 					break;
18812b4ffbf6SChris Packham 				}
18822b4ffbf6SChris Packham 				DEBUG_LEVELING(DEBUG_LEVEL_TRACE, ("\n"));
18832b4ffbf6SChris Packham 			}
18842b4ffbf6SChris Packham 			DEBUG_LEVELING(DEBUG_LEVEL_TRACE, ("pass_lock_num %d\n", pass_lock_num));
18852b4ffbf6SChris Packham 			/* exit condition */
18862b4ffbf6SChris Packham 			if (pass_lock_num == MAX_BUS_NUM)
18872b4ffbf6SChris Packham 				break;
18882b4ffbf6SChris Packham 		} /* for-loop on i */
18892b4ffbf6SChris Packham 
18902b4ffbf6SChris Packham 		if (pass_lock_num != MAX_BUS_NUM) {
18912b4ffbf6SChris Packham 			DEBUG_LEVELING(DEBUG_LEVEL_ERROR,
18922b4ffbf6SChris Packham 				       ("%s: cs %d, pass_lock_num %d, max_bus_num %d, init_pass_lock_num %d\n",
18932b4ffbf6SChris Packham 				       __func__, effective_cs, pass_lock_num, MAX_BUS_NUM, init_pass_lock_num));
18942b4ffbf6SChris Packham 			for (subphy_id = 0; subphy_id < MAX_BUS_NUM; subphy_id++) {
18952b4ffbf6SChris Packham 				VALIDATE_BUS_ACTIVE(tm->bus_act_mask, subphy_id);
18962b4ffbf6SChris Packham 				DEBUG_LEVELING(DEBUG_LEVEL_ERROR,
18972b4ffbf6SChris Packham 					       ("%s: subphy %d %s\n",
18982b4ffbf6SChris Packham 						__func__, subphy_id,
18992b4ffbf6SChris Packham 						(rl_state[effective_cs][subphy_id][if_id] == RL_BEHIND) ?
19002b4ffbf6SChris Packham 						"locked" : "not locked"));
19012b4ffbf6SChris Packham 			}
19022b4ffbf6SChris Packham 		}
19032b4ffbf6SChris Packham 	} /* for-loop on effective_cs */
19042b4ffbf6SChris Packham 
19052b4ffbf6SChris Packham 	/* post-processing read leveling results */
19062b4ffbf6SChris Packham 	if_id = 0;
19072b4ffbf6SChris Packham 	for (effective_cs = 0; effective_cs < max_cs; effective_cs++) {
19082b4ffbf6SChris Packham 		phase_delta = 0;
19092b4ffbf6SChris Packham 		i = rl_min_val[effective_cs];
19102b4ffbf6SChris Packham 		sdr_cycle_incr = i / TAPS_PER_RD_SAMPLE; /* sdr cycle increment */
19112b4ffbf6SChris Packham 		rd_sample = cl_val + 2 * sdr_cycle_incr;
19122b4ffbf6SChris Packham 		rd_ready = rd_sample + RD_FIFO_DLY;
19132b4ffbf6SChris Packham 		min_phase = (rl_min_val[effective_cs] - (sdr_cycle_incr * TAPS_PER_RD_SAMPLE)) % MAX_RD_SAMPLES;
19142b4ffbf6SChris Packham 		max_phase = (rl_max_val[effective_cs] - (sdr_cycle_incr * TAPS_PER_RD_SAMPLE)) % MAX_RD_SAMPLES;
19152b4ffbf6SChris Packham 		final_rd_sample = rd_sample;
19162b4ffbf6SChris Packham 		final_rd_ready = rd_ready;
19172b4ffbf6SChris Packham 
19182b4ffbf6SChris Packham 		ddr3_tip_if_write(0, ACCESS_TYPE_UNICAST, 0, RD_DATA_SMPL_DLYS_REG,
19192b4ffbf6SChris Packham 				  rd_sample << RD_SMPL_DLY_CS_OFFS(effective_cs),
19202b4ffbf6SChris Packham 				  RD_SMPL_DLY_CS_MASK << RD_SMPL_DLY_CS_OFFS(effective_cs));
19212b4ffbf6SChris Packham 		ddr3_tip_if_write(0, ACCESS_TYPE_UNICAST, 0, RD_DATA_RDY_DLYS_REG,
19222b4ffbf6SChris Packham 				  rd_ready << RD_RDY_DLY_CS_OFFS(effective_cs),
19232b4ffbf6SChris Packham 				  RD_RDY_DLY_CS_MASK << RD_RDY_DLY_CS_OFFS(effective_cs));
19242b4ffbf6SChris Packham 		DEBUG_LEVELING(DEBUG_LEVEL_INFO,
19252b4ffbf6SChris Packham 			       ("%s: cs %d, min phase %d, max phase %d, read sample %d\n",
19262b4ffbf6SChris Packham 				__func__, effective_cs, min_phase, max_phase, rd_sample));
19272b4ffbf6SChris Packham 
19282b4ffbf6SChris Packham 		for (subphy_id = 0; subphy_id < MAX_BUS_NUM; subphy_id++) {
19292b4ffbf6SChris Packham 			VALIDATE_BUS_ACTIVE(tm->bus_act_mask, subphy_id);
19302b4ffbf6SChris Packham 			/* reduce sdr cycle per cs; extract rl adll and phase values */
19312b4ffbf6SChris Packham 			i = rl_values[effective_cs][subphy_id][if_id] - (sdr_cycle_incr * TAPS_PER_RD_SAMPLE);
19322b4ffbf6SChris Packham 			rl_adll_val = i % MAX_RD_SAMPLES;
19332b4ffbf6SChris Packham 			rl_phase_val = i / MAX_RD_SAMPLES;
19342b4ffbf6SChris Packham 			rl_phase_val -= phase_delta;
19352b4ffbf6SChris Packham 			DEBUG_LEVELING(DEBUG_LEVEL_INFO,
19362b4ffbf6SChris Packham 				       ("%s: final results: cs %d, subphy %d, read sample %d read ready %d, rl_phase_val %d, rl_adll_val %d\n",
19372b4ffbf6SChris Packham 					__func__, effective_cs, subphy_id, final_rd_sample,
19382b4ffbf6SChris Packham 					final_rd_ready, rl_phase_val, rl_adll_val));
19392b4ffbf6SChris Packham 
19402b4ffbf6SChris Packham 			rl_val = ((rl_adll_val & RL_REF_DLY_MASK) << RL_REF_DLY_OFFS) |
19412b4ffbf6SChris Packham 				 ((rl_phase_val & RL_PH_SEL_MASK) << RL_PH_SEL_OFFS);
19422b4ffbf6SChris Packham 			ddr3_tip_bus_write(dev_num, ACCESS_TYPE_MULTICAST, PARAM_NOT_CARE, ACCESS_TYPE_UNICAST,
19432b4ffbf6SChris Packham 					   subphy_id, subphy_type, RL_PHY_REG(effective_cs), rl_val);
19442b4ffbf6SChris Packham 		}
19452b4ffbf6SChris Packham 	} /* for-loop on effective cs */
19462b4ffbf6SChris Packham 
19472b4ffbf6SChris Packham 	for (if_id = 0; if_id < MAX_INTERFACE_NUM; if_id++) {
19482b4ffbf6SChris Packham 		VALIDATE_IF_ACTIVE(tm->if_act_mask, if_id);
19492b4ffbf6SChris Packham 		if (odt_config != 0)
19502b4ffbf6SChris Packham 			CHECK_STATUS(ddr3_tip_write_additional_odt_setting(dev_num, if_id));
19512b4ffbf6SChris Packham 	}
19522b4ffbf6SChris Packham 
1953*ebb1a593SChris Packham 
19542b4ffbf6SChris Packham 	/* reset read fifo assertion */
19552b4ffbf6SChris Packham 	ddr3_tip_if_write(dev_num, ACCESS_TYPE_MULTICAST, if_id, SDRAM_CFG_REG,
19562b4ffbf6SChris Packham 			  DATA_PUP_RD_RESET_ENA << DATA_PUP_RD_RESET_OFFS,
19572b4ffbf6SChris Packham 			  DATA_PUP_RD_RESET_MASK << DATA_PUP_RD_RESET_OFFS);
19582b4ffbf6SChris Packham 
19592b4ffbf6SChris Packham 	/* reset read fifo deassertion */
19602b4ffbf6SChris Packham 	ddr3_tip_if_write(dev_num, ACCESS_TYPE_MULTICAST, if_id, SDRAM_CFG_REG,
19612b4ffbf6SChris Packham 			  DATA_PUP_RD_RESET_DIS << DATA_PUP_RD_RESET_OFFS,
19622b4ffbf6SChris Packham 			  DATA_PUP_RD_RESET_MASK << DATA_PUP_RD_RESET_OFFS);
19632b4ffbf6SChris Packham 
19642b4ffbf6SChris Packham 	return MV_OK;
19652b4ffbf6SChris Packham }
1966