xref: /openbmc/u-boot/drivers/ddr/marvell/axp/ddr3_dqs.c (revision 344c837686b4268882ee4942f2a1e5e5716c7383)
1 /*
2  * Copyright (C) Marvell International Ltd. and its affiliates
3  *
4  * SPDX-License-Identifier:	GPL-2.0
5  */
6 
7 #include <common.h>
8 #include <i2c.h>
9 #include <spl.h>
10 #include <asm/io.h>
11 #include <asm/arch/cpu.h>
12 #include <asm/arch/soc.h>
13 
14 #include "ddr3_hw_training.h"
15 
16 /*
17  * Debug
18  */
19 #define DEBUG_DQS_C(s, d, l) \
20 	DEBUG_DQS_S(s); DEBUG_DQS_D(d, l); DEBUG_DQS_S("\n")
21 #define DEBUG_DQS_FULL_C(s, d, l) \
22 	DEBUG_DQS_FULL_S(s); DEBUG_DQS_FULL_D(d, l); DEBUG_DQS_FULL_S("\n")
23 #define DEBUG_DQS_RESULTS_C(s, d, l) \
24 	DEBUG_DQS_RESULTS_S(s); DEBUG_DQS_RESULTS_D(d, l); DEBUG_DQS_RESULTS_S("\n")
25 #define DEBUG_PER_DQ_C(s, d, l) \
26 	puts(s); printf("%x", d); puts("\n")
27 
28 #define DEBUG_DQS_RESULTS_S(s) \
29 	debug_cond(ddr3_get_log_level() >= MV_LOG_LEVEL_2, "%s", s)
30 #define DEBUG_DQS_RESULTS_D(d, l) \
31 	debug_cond(ddr3_get_log_level() >= MV_LOG_LEVEL_2, "%x", d)
32 
33 #define DEBUG_PER_DQ_S(s) \
34 	debug_cond(ddr3_get_log_level() >= MV_LOG_LEVEL_3, "%s", s)
35 #define DEBUG_PER_DQ_D(d, l) \
36 	debug_cond(ddr3_get_log_level() >= MV_LOG_LEVEL_3, "%x", d)
37 #define DEBUG_PER_DQ_DD(d, l) \
38 	debug_cond(ddr3_get_log_level() >= MV_LOG_LEVEL_3, "%d", d)
39 
40 #ifdef MV_DEBUG_DQS
41 #define DEBUG_DQS_S(s)			puts(s)
42 #define DEBUG_DQS_D(d, l)		printf("%x", d)
43 #else
44 #define DEBUG_DQS_S(s)
45 #define DEBUG_DQS_D(d, l)
46 #endif
47 
48 #ifdef MV_DEBUG_DQS_FULL
49 #define DEBUG_DQS_FULL_S(s)		puts(s)
50 #define DEBUG_DQS_FULL_D(d, l)		printf("%x", d)
51 #else
52 #define DEBUG_DQS_FULL_S(s)
53 #define DEBUG_DQS_FULL_D(d, l)
54 #endif
55 
56 /* State machine for centralization - find low & high limit */
57 enum {
58 	PUP_ADLL_LIMITS_STATE_FAIL,
59 	PUP_ADLL_LIMITS_STATE_PASS,
60 	PUP_ADLL_LIMITS_STATE_FAIL_AFTER_PASS,
61 };
62 
63 /* Hold centralization low results */
64 static int centralization_low_limit[MAX_PUP_NUM] = { 0 };
65 /* Hold centralization high results */
66 static int centralization_high_limit[MAX_PUP_NUM] = { 0 };
67 
68 int ddr3_find_adll_limits(MV_DRAM_INFO *dram_info, u32 cs, u32 ecc, int is_tx);
69 int ddr3_check_window_limits(u32 pup, int high_limit, int low_limit, int is_tx,
70 			  int *size_valid);
71 static int ddr3_center_calc(MV_DRAM_INFO *dram_info, u32 cs, u32 ecc,
72 			    int is_tx);
73 int ddr3_special_pattern_i_search(MV_DRAM_INFO *dram_info, u32 cs, u32 ecc,
74 			      int is_tx, u32 special_pattern_pup);
75 int ddr3_special_pattern_ii_search(MV_DRAM_INFO *dram_info, u32 cs, u32 ecc,
76 				   int is_tx, u32 special_pattern_pup);
77 int ddr3_set_dqs_centralization_results(MV_DRAM_INFO *dram_info, u32 cs, u32 ecc,
78 				    int is_tx);
79 
80 #ifdef MV88F78X60
81 extern u32 killer_pattern_32b[DQ_NUM][LEN_SPECIAL_PATTERN];
82 extern u32 killer_pattern_64b[DQ_NUM][LEN_SPECIAL_PATTERN];
83 extern int per_bit_data[MAX_PUP_NUM][DQ_NUM];
84 #else
85 extern u32 killer_pattern[DQ_NUM][LEN_16BIT_KILLER_PATTERN];
86 extern u32 killer_pattern_32b[DQ_NUM][LEN_SPECIAL_PATTERN];
87 #if defined(MV88F672X)
88 extern int per_bit_data[MAX_PUP_NUM][DQ_NUM];
89 #endif
90 #endif
91 extern u32 special_pattern[DQ_NUM][LEN_SPECIAL_PATTERN];
92 
93 static u32 *ddr3_dqs_choose_pattern(MV_DRAM_INFO *dram_info, u32 victim_dq)
94 {
95 	u32 *pattern_ptr;
96 
97 	/* Choose pattern */
98 	switch (dram_info->ddr_width) {
99 #if defined(MV88F672X)
100 	case 16:
101 		pattern_ptr = (u32 *)&killer_pattern[victim_dq];
102 		break;
103 #endif
104 	case 32:
105 		pattern_ptr = (u32 *)&killer_pattern_32b[victim_dq];
106 		break;
107 #if defined(MV88F78X60)
108 	case 64:
109 		pattern_ptr = (u32 *)&killer_pattern_64b[victim_dq];
110 		break;
111 #endif
112 	default:
113 #if defined(MV88F78X60)
114 		pattern_ptr = (u32 *)&killer_pattern_32b[victim_dq];
115 #else
116 		pattern_ptr = (u32 *)&killer_pattern[victim_dq];
117 #endif
118 		break;
119 	}
120 
121 	return pattern_ptr;
122 }
123 
124 /*
125  * Name:     ddr3_dqs_centralization_rx
126  * Desc:     Execute the DQS centralization RX phase.
127  * Args:     dram_info
128  * Notes:
129  * Returns:  MV_OK if success, other error code if fail.
130  */
131 int ddr3_dqs_centralization_rx(MV_DRAM_INFO *dram_info)
132 {
133 	u32 cs, ecc, reg;
134 	int status;
135 
136 	DEBUG_DQS_S("DDR3 - DQS Centralization RX - Starting procedure\n");
137 
138 	/* Enable SW override */
139 	reg = reg_read(REG_DRAM_TRAINING_2_ADDR) |
140 		(1 << REG_DRAM_TRAINING_2_SW_OVRD_OFFS);
141 
142 	/* [0] = 1 - Enable SW override  */
143 	/* 0x15B8 - Training SW 2 Register */
144 	reg_write(REG_DRAM_TRAINING_2_ADDR, reg);
145 	DEBUG_DQS_S("DDR3 - DQS Centralization RX - SW Override Enabled\n");
146 
147 	reg = (1 << REG_DRAM_TRAINING_AUTO_OFFS);
148 	reg_write(REG_DRAM_TRAINING_ADDR, reg);	/* 0x15B0 - Training Register */
149 
150 	/* Loop for each CS */
151 	for (cs = 0; cs < MAX_CS; cs++) {
152 		if (dram_info->cs_ena & (1 << cs)) {
153 			DEBUG_DQS_FULL_C("DDR3 - DQS Centralization RX - CS - ",
154 					 (u32) cs, 1);
155 
156 			for (ecc = 0; ecc < (dram_info->ecc_ena + 1); ecc++) {
157 
158 				/* ECC Support - Switch ECC Mux on ecc=1 */
159 				reg = reg_read(REG_DRAM_TRAINING_2_ADDR) &
160 					~(1 << REG_DRAM_TRAINING_2_ECC_MUX_OFFS);
161 				reg |= (dram_info->ecc_ena *
162 					ecc << REG_DRAM_TRAINING_2_ECC_MUX_OFFS);
163 				reg_write(REG_DRAM_TRAINING_2_ADDR, reg);
164 
165 				if (ecc)
166 					DEBUG_DQS_FULL_S("DDR3 - DQS Centralization RX - ECC Mux Enabled\n");
167 				else
168 					DEBUG_DQS_FULL_S("DDR3 - DQS Centralization RX - ECC Mux Disabled\n");
169 
170 				DEBUG_DQS_FULL_S("DDR3 - DQS Centralization RX - Find all limits\n");
171 
172 				status = ddr3_find_adll_limits(dram_info, cs,
173 							       ecc, 0);
174 				if (MV_OK != status)
175 					return status;
176 
177 				DEBUG_DQS_FULL_S("DDR3 - DQS Centralization RX - Start calculating center\n");
178 
179 				status = ddr3_center_calc(dram_info, cs, ecc,
180 							  0);
181 				if (MV_OK != status)
182 					return status;
183 			}
184 		}
185 	}
186 
187 	/* ECC Support - Disable ECC MUX */
188 	reg = reg_read(REG_DRAM_TRAINING_2_ADDR) &
189 		~(1 << REG_DRAM_TRAINING_2_ECC_MUX_OFFS);
190 	reg_write(REG_DRAM_TRAINING_2_ADDR, reg);
191 
192 	/* Disable SW override - Must be in a different stage */
193 	/* [0]=0 - Enable SW override  */
194 	reg = reg_read(REG_DRAM_TRAINING_2_ADDR);
195 	reg &= ~(1 << REG_DRAM_TRAINING_2_SW_OVRD_OFFS);
196 	/* 0x15B8 - Training SW 2 Register */
197 	reg_write(REG_DRAM_TRAINING_2_ADDR, reg);
198 
199 	reg = reg_read(REG_DRAM_TRAINING_1_ADDR) |
200 		(1 << REG_DRAM_TRAINING_1_TRNBPOINT_OFFS);
201 	reg_write(REG_DRAM_TRAINING_1_ADDR, reg);
202 
203 	return MV_OK;
204 }
205 
206 /*
207  * Name:     ddr3_dqs_centralization_tx
208  * Desc:     Execute the DQS centralization TX phase.
209  * Args:     dram_info
210  * Notes:
211  * Returns:  MV_OK if success, other error code if fail.
212  */
213 int ddr3_dqs_centralization_tx(MV_DRAM_INFO *dram_info)
214 {
215 	u32 cs, ecc, reg;
216 	int status;
217 
218 	DEBUG_DQS_S("DDR3 - DQS Centralization TX - Starting procedure\n");
219 
220 	/* Enable SW override */
221 	reg = reg_read(REG_DRAM_TRAINING_2_ADDR) |
222 		(1 << REG_DRAM_TRAINING_2_SW_OVRD_OFFS);
223 
224 	/* [0] = 1 - Enable SW override  */
225 	/* 0x15B8 - Training SW 2 Register */
226 	reg_write(REG_DRAM_TRAINING_2_ADDR, reg);
227 	DEBUG_DQS_S("DDR3 - DQS Centralization TX - SW Override Enabled\n");
228 
229 	reg = (1 << REG_DRAM_TRAINING_AUTO_OFFS);
230 	reg_write(REG_DRAM_TRAINING_ADDR, reg);	/* 0x15B0 - Training Register */
231 
232 	/* Loop for each CS */
233 	for (cs = 0; cs < MAX_CS; cs++) {
234 		if (dram_info->cs_ena & (1 << cs)) {
235 			DEBUG_DQS_FULL_C("DDR3 - DQS Centralization TX - CS - ",
236 					 (u32) cs, 1);
237 			for (ecc = 0; ecc < (dram_info->ecc_ena + 1); ecc++) {
238 				/* ECC Support - Switch ECC Mux on ecc=1 */
239 				reg = reg_read(REG_DRAM_TRAINING_2_ADDR) &
240 					~(1 << REG_DRAM_TRAINING_2_ECC_MUX_OFFS);
241 				reg |= (dram_info->ecc_ena *
242 					ecc << REG_DRAM_TRAINING_2_ECC_MUX_OFFS);
243 				reg_write(REG_DRAM_TRAINING_2_ADDR, reg);
244 
245 				if (ecc)
246 					DEBUG_DQS_FULL_S("DDR3 - DQS Centralization TX - ECC Mux Enabled\n");
247 				else
248 					DEBUG_DQS_FULL_S("DDR3 - DQS Centralization TX - ECC Mux Disabled\n");
249 
250 				DEBUG_DQS_FULL_S("DDR3 - DQS Centralization TX - Find all limits\n");
251 
252 				status = ddr3_find_adll_limits(dram_info, cs,
253 							       ecc, 1);
254 				if (MV_OK != status)
255 					return status;
256 
257 				DEBUG_DQS_FULL_S("DDR3 - DQS Centralization TX - Start calculating center\n");
258 
259 				status = ddr3_center_calc(dram_info, cs, ecc,
260 							  1);
261 				if (MV_OK != status)
262 					return status;
263 			}
264 		}
265 	}
266 
267 	/* ECC Support - Disable ECC MUX */
268 	reg = reg_read(REG_DRAM_TRAINING_2_ADDR) &
269 		~(1 << REG_DRAM_TRAINING_2_ECC_MUX_OFFS);
270 	reg_write(REG_DRAM_TRAINING_2_ADDR, reg);
271 
272 	/* Disable SW override - Must be in a different stage */
273 	/* [0]=0 - Enable SW override  */
274 	reg = reg_read(REG_DRAM_TRAINING_2_ADDR);
275 	reg &= ~(1 << REG_DRAM_TRAINING_2_SW_OVRD_OFFS);
276 	/* 0x15B8 - Training SW 2 Register */
277 	reg_write(REG_DRAM_TRAINING_2_ADDR, reg);
278 
279 	reg = reg_read(REG_DRAM_TRAINING_1_ADDR) |
280 		(1 << REG_DRAM_TRAINING_1_TRNBPOINT_OFFS);
281 	reg_write(REG_DRAM_TRAINING_1_ADDR, reg);
282 
283 	return MV_OK;
284 }
285 
286 /*
287  * Name:     ddr3_find_adll_limits
288  * Desc:     Execute the Find ADLL limits phase.
289  * Args:     dram_info
290  *           cs
291  *           ecc_ena
292  *           is_tx             Indicate whether Rx or Tx
293  * Notes:
294  * Returns:  MV_OK if success, other error code if fail.
295  */
296 int ddr3_find_adll_limits(MV_DRAM_INFO *dram_info, u32 cs, u32 ecc, int is_tx)
297 {
298 	u32 victim_dq, pup, tmp;
299 	u32 adll_addr;
300 	u32 max_pup;		/* maximal pup index */
301 	u32 pup_mask = 0;
302 	u32 unlock_pup;		/* bit array of un locked pups */
303 	u32 new_unlock_pup;	/* bit array of compare failed pups */
304 	u32 curr_adll;
305 	u32 adll_start_val;	/* adll start loop value - for rx or tx limit */
306 	u32 high_limit;	/* holds found High Limit */
307 	u32 low_limit;		/* holds found Low Limit */
308 	int win_valid;
309 	int update_win;
310 	u32 sdram_offset;
311 	u32 uj, cs_count, cs_tmp, ii;
312 	u32 *pattern_ptr;
313 	u32 dq;
314 	u32 adll_end_val;	/* adll end of loop val - for rx or tx limit */
315 	u8 analog_pbs[DQ_NUM][MAX_PUP_NUM][DQ_NUM][2];
316 	u8 analog_pbs_sum[MAX_PUP_NUM][DQ_NUM][2];
317 	int pup_adll_limit_state[MAX_PUP_NUM];	/* hold state of each pup */
318 
319 	adll_addr = ((is_tx == 1) ? PUP_DQS_WR : PUP_DQS_RD);
320 	adll_end_val = ((is_tx == 1) ? ADLL_MIN : ADLL_MAX);
321 	adll_start_val = ((is_tx == 1) ? ADLL_MAX : ADLL_MIN);
322 	max_pup = (ecc + (1 - ecc) * dram_info->num_of_std_pups);
323 
324 	DEBUG_DQS_FULL_S("DDR3 - DQS Find Limits - Starting Find ADLL Limits\n");
325 
326 	/* init the array */
327 	for (pup = 0; pup < max_pup; pup++) {
328 		centralization_low_limit[pup] = ADLL_MIN;
329 		centralization_high_limit[pup] = ADLL_MAX;
330 	}
331 
332 	/* Killer Pattern */
333 	cs_count = 0;
334 	for (cs_tmp = 0; cs_tmp < cs; cs_tmp++) {
335 		if (dram_info->cs_ena & (1 << cs_tmp))
336 			cs_count++;
337 	}
338 	sdram_offset = cs_count * (SDRAM_CS_SIZE + 1);
339 	sdram_offset += ((is_tx == 1) ?
340 			 SDRAM_DQS_TX_OFFS : SDRAM_DQS_RX_OFFS);
341 
342 	/* Prepare pup masks */
343 	for (pup = 0; pup < max_pup; pup++)
344 		pup_mask |= (1 << pup);
345 
346 	for (pup = 0; pup < max_pup; pup++) {
347 		for (dq = 0; dq < DQ_NUM; dq++) {
348 			analog_pbs_sum[pup][dq][0] = adll_start_val;
349 			analog_pbs_sum[pup][dq][1] = adll_end_val;
350 		}
351 	}
352 
353 	/* Loop - use different pattern for each victim_dq */
354 	for (victim_dq = 0; victim_dq < DQ_NUM; victim_dq++) {
355 		DEBUG_DQS_FULL_C("DDR3 - DQS Find Limits - Victim DQ - ",
356 				 (u32)victim_dq, 1);
357 		/*
358 		 * The pups 3 bit arrays represent state machine. with
359 		 * 3 stages for each pup.
360 		 * 1. fail and didn't get pass in earlier compares.
361 		 * 2. pass compare
362 		 * 3. fail after pass - end state.
363 		 * The window limits are the adll values where the adll
364 		 * was in the pass stage.
365 		 */
366 
367 		/* Set all states to Fail (1st state) */
368 		for (pup = 0; pup < max_pup; pup++)
369 			pup_adll_limit_state[pup] = PUP_ADLL_LIMITS_STATE_FAIL;
370 
371 		/* Set current valid pups */
372 		unlock_pup = pup_mask;
373 
374 		/* Set ADLL to start value */
375 		curr_adll = adll_start_val;
376 
377 #if defined(MV88F78X60)
378 		for (pup = 0; pup < max_pup; pup++) {
379 			for (dq = 0; dq < DQ_NUM; dq++) {
380 				analog_pbs[victim_dq][pup][dq][0] =
381 					adll_start_val;
382 				analog_pbs[victim_dq][pup][dq][1] =
383 					adll_end_val;
384 				per_bit_data[pup][dq] = 0;
385 			}
386 		}
387 #endif
388 
389 		for (uj = 0; uj < ADLL_MAX; uj++) {
390 			DEBUG_DQS_FULL_C("DDR3 - DQS Find Limits - Setting ADLL to ",
391 					 curr_adll, 2);
392 			for (pup = 0; pup < max_pup; pup++) {
393 				if (IS_PUP_ACTIVE(unlock_pup, pup) == 1) {
394 					tmp = ((is_tx == 1) ? curr_adll +
395 					       dram_info->wl_val[cs]
396 					       [pup * (1 - ecc) + ecc * ECC_PUP]
397 					       [D] : curr_adll);
398 					ddr3_write_pup_reg(adll_addr, cs, pup +
399 						(ecc * ECC_PUP), 0, tmp);
400 				}
401 			}
402 
403 			/* Choose pattern */
404 			pattern_ptr = ddr3_dqs_choose_pattern(dram_info,
405 							      victim_dq);
406 
407 			/* '1' - means pup failed, '0' - means pup pass */
408 			new_unlock_pup = 0;
409 
410 			/* Read and compare results for Victim_DQ# */
411 			for (ii = 0; ii < 3; ii++) {
412 				u32 tmp = 0;
413 				if (MV_OK != ddr3_sdram_dqs_compare(dram_info,
414 							   unlock_pup, &tmp,
415 							   pattern_ptr,
416 							   LEN_KILLER_PATTERN,
417 							   sdram_offset +
418 							   LEN_KILLER_PATTERN *
419 							   4 * victim_dq,
420 							   is_tx, 0, NULL,
421 							   0))
422 					return MV_DDR3_TRAINING_ERR_DRAM_COMPARE;
423 
424 				new_unlock_pup |= tmp;
425 			}
426 
427 			pup = 0;
428 			DEBUG_DQS_FULL_C("DDR3 - DQS Find Limits - UnlockPup: ",
429 					 unlock_pup, 2);
430 			DEBUG_DQS_FULL_C("DDR3 - DQS Find Limits - NewUnlockPup: ",
431 					 new_unlock_pup, 2);
432 
433 			/* Update pup state */
434 			for (pup = 0; pup < max_pup; pup++) {
435 				if (IS_PUP_ACTIVE(unlock_pup, pup) == 0) {
436 					DEBUG_DQS_FULL_C("DDR3 - DQS Find Limits - Skipping pup ",
437 							 pup, 1);
438 					continue;
439 				}
440 
441 				/*
442 				 * Still didn't find the window limit of the pup
443 				 */
444 				if (IS_PUP_ACTIVE(new_unlock_pup, pup) == 1) {
445 					/* Current compare result == fail */
446 					if (pup_adll_limit_state[pup] ==
447 					    PUP_ADLL_LIMITS_STATE_PASS) {
448 						/*
449 						 * If now it failed but passed
450 						 * earlier
451 						 */
452 						DEBUG_DQS_S("DDR3 - DQS Find Limits - PASS to FAIL: CS - ");
453 						DEBUG_DQS_D(cs, 1);
454 						DEBUG_DQS_S(", DQ - ");
455 						DEBUG_DQS_D(victim_dq, 1);
456 						DEBUG_DQS_S(", Pup - ");
457 						DEBUG_DQS_D(pup, 1);
458 						DEBUG_DQS_S(", ADLL - ");
459 						DEBUG_DQS_D(curr_adll, 2);
460 						DEBUG_DQS_S("\n");
461 
462 #if defined(MV88F78X60)
463 						for (dq = 0; dq < DQ_NUM; dq++) {
464 							if ((analog_pbs[victim_dq][pup][dq][0] != adll_start_val)
465 							    && (analog_pbs[victim_dq][pup]
466 								[dq][1] == adll_end_val))
467 								analog_pbs
468 									[victim_dq]
469 									[pup][dq]
470 									[1] =
471 									curr_adll;
472 						}
473 #endif
474 						win_valid = 1;
475 						update_win = 0;
476 
477 						/* Keep min / max limit value */
478 						if (is_tx == 0) {
479 							/* RX - found upper limit */
480 							if (centralization_high_limit[pup] >
481 							    (curr_adll - 1)) {
482 								high_limit =
483 									curr_adll - 1;
484 								low_limit =
485 									centralization_low_limit[pup];
486 								update_win = 1;
487 							}
488 						} else {
489 							/* TX - found lower limit */
490 							if (centralization_low_limit[pup] < (curr_adll + 1)) {
491 								high_limit =
492 									centralization_high_limit
493 									[pup];
494 								low_limit =
495 									curr_adll + 1;
496 								update_win =
497 									1;
498 							}
499 						}
500 
501 						if (update_win == 1) {
502 							/*
503 							 * Before updating
504 							 * window limits we need
505 							 * to check that the
506 							 * limits are valid
507 							 */
508 							if (MV_OK !=
509 							    ddr3_check_window_limits
510 							    (pup, high_limit,
511 							     low_limit, is_tx,
512 							     &win_valid))
513 								return MV_DDR3_TRAINING_ERR_WIN_LIMITS;
514 
515 							if (win_valid == 1) {
516 								/*
517 								 * Window limits
518 								 * should be
519 								 * updated
520 								 */
521 								centralization_low_limit
522 									[pup] =
523 									low_limit;
524 								centralization_high_limit
525 									[pup] =
526 									high_limit;
527 							}
528 						}
529 
530 						if (win_valid == 1) {
531 							/* Found end of window - lock the pup */
532 							pup_adll_limit_state[pup] =
533 								PUP_ADLL_LIMITS_STATE_FAIL_AFTER_PASS;
534 							unlock_pup &= ~(1 << pup);
535 						} else {
536 							/* Probably false pass - reset status */
537 							pup_adll_limit_state[pup] =
538 								PUP_ADLL_LIMITS_STATE_FAIL;
539 
540 #if defined(MV88F78X60)
541 							/* Clear logging array of win size (per Dq) */
542 							for (dq = 0;
543 							     dq < DQ_NUM;
544 							     dq++) {
545 								analog_pbs
546 									[victim_dq]
547 									[pup][dq]
548 									[0] =
549 									adll_start_val;
550 								analog_pbs
551 									[victim_dq]
552 									[pup][dq]
553 									[1] =
554 									adll_end_val;
555 								per_bit_data
556 									[pup][dq]
557 									= 0;
558 							}
559 #endif
560 						}
561 					}
562 				} else {
563 					/* Current compare result == pass */
564 					if (pup_adll_limit_state[pup] ==
565 					    PUP_ADLL_LIMITS_STATE_FAIL) {
566 						/* If now it passed but failed earlier */
567 						DEBUG_DQS_S("DDR3 - DQS Find Limits - FAIL to PASS: CS - ");
568 						DEBUG_DQS_D(cs, 1);
569 						DEBUG_DQS_S(", DQ - ");
570 						DEBUG_DQS_D(victim_dq, 1);
571 						DEBUG_DQS_S(", Pup - ");
572 						DEBUG_DQS_D(pup, 1);
573 						DEBUG_DQS_S(", ADLL - ");
574 						DEBUG_DQS_D(curr_adll, 2);
575 						DEBUG_DQS_S("\n");
576 
577 #if defined(MV88F78X60)
578 						for (dq = 0; dq < DQ_NUM;
579 						     dq++) {
580 							if (analog_pbs[victim_dq][pup][dq][0] == adll_start_val)
581 								analog_pbs
582 								    [victim_dq]
583 								    [pup][dq]
584 								    [0] =
585 								    curr_adll;
586 						}
587 #endif
588 						/* Found start of window */
589 						pup_adll_limit_state[pup] =
590 						    PUP_ADLL_LIMITS_STATE_PASS;
591 
592 						/* Keep min / max limit value */
593 						if (is_tx == 0) {
594 							/* RX - found low limit */
595 							if (centralization_low_limit[pup] <= curr_adll)
596 								centralization_low_limit
597 								    [pup] =
598 								    curr_adll;
599 						} else {
600 							/* TX - found high limit */
601 							if (centralization_high_limit[pup] >= curr_adll)
602 								centralization_high_limit
603 								    [pup] =
604 								    curr_adll;
605 						}
606 					}
607 				}
608 			}
609 
610 			if (unlock_pup == 0) {
611 				/* Found limit to all pups */
612 				DEBUG_DQS_FULL_S("DDR3 - DQS Find Limits - found PUP limit\n");
613 				break;
614 			}
615 
616 			/*
617 			 * Increment / decrement (Move to right / left
618 			 * one phase - ADLL) dqs RX / TX delay (for all un
619 			 * lock pups
620 			 */
621 			if (is_tx == 0)
622 				curr_adll++;
623 			else
624 				curr_adll--;
625 		}
626 
627 		if (unlock_pup != 0) {
628 			/*
629 			 * Found pups that didn't reach to the end of the
630 			 * state machine
631 			 */
632 			DEBUG_DQS_C("DDR3 - DQS Find Limits - Pups that didn't reached end of the state machine: ",
633 				    unlock_pup, 1);
634 
635 			for (pup = 0; pup < max_pup; pup++) {
636 				if (IS_PUP_ACTIVE(unlock_pup, pup) == 1) {
637 					if (pup_adll_limit_state[pup] ==
638 					    PUP_ADLL_LIMITS_STATE_FAIL) {
639 						/* ERROR - found fail for all window size */
640 						DEBUG_DQS_S("DDR3 - DQS Find Limits - Got FAIL for the complete range on pup - ");
641 						DEBUG_DQS_D(pup, 1);
642 						DEBUG_DQS_C(" victim DQ ",
643 							    victim_dq, 1);
644 
645 						/* For debug - set min limit to illegal limit */
646 						centralization_low_limit[pup]
647 							= ADLL_ERROR;
648 						/*
649 						 * In case the pup is in mode
650 						 * PASS - the limit is the min
651 						 * / max adll, no need to
652 						 * update because of the results
653 						 * array default value
654 						 */
655 						return MV_DDR3_TRAINING_ERR_PUP_RANGE;
656 					}
657 				}
658 			}
659 		}
660 	}
661 
662 	DEBUG_DQS_S("DDR3 - DQS Find Limits - DQ values per victim results:\n");
663 	for (victim_dq = 0; victim_dq < DQ_NUM; victim_dq++) {
664 		for (pup = 0; pup < max_pup; pup++) {
665 			DEBUG_DQS_S("Victim DQ-");
666 			DEBUG_DQS_D(victim_dq, 1);
667 			DEBUG_DQS_S(", PUP-");
668 			DEBUG_DQS_D(pup, 1);
669 			for (dq = 0; dq < DQ_NUM; dq++) {
670 				DEBUG_DQS_S(", DQ-");
671 				DEBUG_DQS_D(dq, 1);
672 				DEBUG_DQS_S(",S-");
673 				DEBUG_DQS_D(analog_pbs[victim_dq][pup][dq]
674 					    [0], 2);
675 				DEBUG_DQS_S(",E-");
676 				DEBUG_DQS_D(analog_pbs[victim_dq][pup][dq]
677 					    [1], 2);
678 
679 				if (is_tx == 0) {
680 					if (analog_pbs[victim_dq][pup][dq][0]
681 					    > analog_pbs_sum[pup][dq][0])
682 						analog_pbs_sum[pup][dq][0] =
683 						    analog_pbs[victim_dq][pup]
684 						    [dq][0];
685 					if (analog_pbs[victim_dq][pup][dq][1]
686 					    < analog_pbs_sum[pup][dq][1])
687 						analog_pbs_sum[pup][dq][1] =
688 						    analog_pbs[victim_dq][pup]
689 						    [dq][1];
690 				} else {
691 					if (analog_pbs[victim_dq][pup][dq][0]
692 					    < analog_pbs_sum[pup][dq][0])
693 						analog_pbs_sum[pup][dq][0] =
694 						    analog_pbs[victim_dq][pup]
695 						    [dq][0];
696 					if (analog_pbs[victim_dq][pup][dq][1]
697 					    > analog_pbs_sum[pup][dq][1])
698 						analog_pbs_sum[pup][dq][1] =
699 						    analog_pbs[victim_dq][pup]
700 						    [dq][1];
701 				}
702 			}
703 			DEBUG_DQS_S("\n");
704 		}
705 	}
706 
707 	if (ddr3_get_log_level() >= MV_LOG_LEVEL_3) {
708 		u32 dq;
709 
710 		DEBUG_PER_DQ_S("\n########## LOG LEVEL 3(Windows margins per-DQ) ##########\n");
711 		if (is_tx) {
712 			DEBUG_PER_DQ_C("DDR3 - TX  CS: ", cs, 1);
713 		} else {
714 			DEBUG_PER_DQ_C("DDR3 - RX  CS: ", cs, 1);
715 		}
716 
717 		if (ecc == 0) {
718 			DEBUG_PER_DQ_S("\n DATA RESULTS:\n");
719 		} else {
720 			DEBUG_PER_DQ_S("\n ECC RESULTS:\n");
721 		}
722 
723 		/* Since all dq has the same value we take 0 as representive */
724 		dq = 0;
725 		for (pup = 0; pup < max_pup; pup++) {
726 			if (ecc == 0) {
727 				DEBUG_PER_DQ_S("\nBYTE:");
728 				DEBUG_PER_DQ_D(pup, 1);
729 				DEBUG_PER_DQ_S("\n");
730 			} else {
731 				DEBUG_PER_DQ_S("\nECC BYTE:\n");
732 			}
733 			DEBUG_PER_DQ_S("  DQ's        LOW       HIGH       WIN-SIZE\n");
734 			DEBUG_PER_DQ_S("============================================\n");
735 			for (victim_dq = 0; victim_dq < DQ_NUM; victim_dq++) {
736 				if (ecc == 0) {
737 					DEBUG_PER_DQ_S("DQ[");
738 					DEBUG_PER_DQ_DD((victim_dq +
739 							 DQ_NUM * pup), 2);
740 					DEBUG_PER_DQ_S("]");
741 				} else {
742 					DEBUG_PER_DQ_S("CB[");
743 					DEBUG_PER_DQ_DD(victim_dq, 2);
744 					DEBUG_PER_DQ_S("]");
745 				}
746 				if (is_tx) {
747 					DEBUG_PER_DQ_S("      0x");
748 					DEBUG_PER_DQ_D(analog_pbs[victim_dq][pup][dq][1], 2);	/* low value */
749 					DEBUG_PER_DQ_S("        0x");
750 					DEBUG_PER_DQ_D(analog_pbs[victim_dq][pup][dq][0], 2);	/* high value */
751 					DEBUG_PER_DQ_S("        0x");
752 					DEBUG_PER_DQ_D(analog_pbs[victim_dq][pup][dq][0] - analog_pbs[victim_dq][pup][dq][1], 2);	/* win-size */
753 				} else {
754 					DEBUG_PER_DQ_S("     0x");
755 					DEBUG_PER_DQ_D(analog_pbs[victim_dq][pup][dq][0], 2);	/* low value */
756 					DEBUG_PER_DQ_S("       0x");
757 					DEBUG_PER_DQ_D((analog_pbs[victim_dq][pup][dq][1] - 1), 2);	/* high value */
758 					DEBUG_PER_DQ_S("       0x");
759 					DEBUG_PER_DQ_D(analog_pbs[victim_dq][pup][dq][1] - analog_pbs[victim_dq][pup][dq][0], 2);	/* win-size */
760 				}
761 				DEBUG_PER_DQ_S("\n");
762 			}
763 		}
764 		DEBUG_PER_DQ_S("\n");
765 	}
766 
767 	if (is_tx) {
768 		DEBUG_DQS_S("DDR3 - DQS TX - Find Limits - DQ values Summary:\n");
769 	} else {
770 		DEBUG_DQS_S("DDR3 - DQS RX - Find Limits - DQ values Summary:\n");
771 	}
772 
773 	for (pup = 0; pup < max_pup; pup++) {
774 		DEBUG_DQS_S("PUP-");
775 		DEBUG_DQS_D(pup, 1);
776 		for (dq = 0; dq < DQ_NUM; dq++) {
777 			DEBUG_DQS_S(", DQ-");
778 			DEBUG_DQS_D(dq, 1);
779 			DEBUG_DQS_S(",S-");
780 			DEBUG_DQS_D(analog_pbs_sum[pup][dq][0], 2);
781 			DEBUG_DQS_S(",E-");
782 			DEBUG_DQS_D(analog_pbs_sum[pup][dq][1], 2);
783 		}
784 		DEBUG_DQS_S("\n");
785 	}
786 
787 	if (is_tx) {
788 		DEBUG_DQS_S("DDR3 - DQS TX - Find Limits - DQ values Summary:\n");
789 	} else {
790 		DEBUG_DQS_S("DDR3 - DQS RX - Find Limits - DQ values Summary:\n");
791 	}
792 
793 	for (pup = 0; pup < max_pup; pup++) {
794 		if (max_pup == 1) {
795 			/* For ECC PUP */
796 			DEBUG_DQS_S("DDR3 - DQS8");
797 		} else {
798 			DEBUG_DQS_S("DDR3 - DQS");
799 			DEBUG_DQS_D(pup, 1);
800 		}
801 
802 		for (dq = 0; dq < DQ_NUM; dq++) {
803 			DEBUG_DQS_S(", DQ-");
804 			DEBUG_DQS_D(dq, 1);
805 			DEBUG_DQS_S("::S-");
806 			DEBUG_DQS_D(analog_pbs_sum[pup][dq][0], 2);
807 			DEBUG_DQS_S(",E-");
808 			DEBUG_DQS_D(analog_pbs_sum[pup][dq][1], 2);
809 		}
810 		DEBUG_DQS_S("\n");
811 	}
812 
813 	DEBUG_DQS_S("DDR3 - DQS Find Limits - Ended\n");
814 
815 	return MV_OK;
816 }
817 
818 /*
819  * Name:     ddr3_check_window_limits
820  * Desc:     Check window High & Low limits.
821  * Args:     pup                pup index
822  *           high_limit           window high limit
823  *           low_limit            window low limit
824  *           is_tx                Indicate whether Rx or Tx
825  *           size_valid          Indicate whether window size is valid
826  * Notes:
827  * Returns:  MV_OK if success, other error code if fail.
828  */
829 int ddr3_check_window_limits(u32 pup, int high_limit, int low_limit, int is_tx,
830 			     int *size_valid)
831 {
832 	DEBUG_DQS_FULL_S("DDR3 - DQS Check Win Limits - Starting\n");
833 
834 	if (low_limit > high_limit) {
835 		DEBUG_DQS_S("DDR3 - DQS Check Win Limits - Pup ");
836 		DEBUG_DQS_D(pup, 1);
837 		DEBUG_DQS_S(" Low Limit grater than High Limit\n");
838 		*size_valid = 0;
839 		return MV_OK;
840 	}
841 
842 	/*
843 	 * Check that window size is valid, if not it was probably false pass
844 	 * before
845 	 */
846 	if ((high_limit - low_limit) < MIN_WIN_SIZE) {
847 		/*
848 		 * Since window size is too small probably there was false
849 		 * pass
850 		 */
851 		*size_valid = 0;
852 
853 		DEBUG_DQS_S("DDR3 - DQS Check Win Limits - Pup ");
854 		DEBUG_DQS_D(pup, 1);
855 		DEBUG_DQS_S(" Window size is smaller than MIN_WIN_SIZE\n");
856 
857 	} else if ((high_limit - low_limit) > ADLL_MAX) {
858 		*size_valid = 0;
859 
860 		DEBUG_DQS_S("DDR3 - DQS Check Win Limits - Pup ");
861 		DEBUG_DQS_D(pup, 1);
862 		DEBUG_DQS_S
863 		    (" Window size is bigger than max ADLL taps (31)  Exiting.\n");
864 
865 		return MV_FAIL;
866 
867 	} else {
868 		*size_valid = 1;
869 
870 		DEBUG_DQS_FULL_S("DDR3 - DQS Check Win Limits - Pup ");
871 		DEBUG_DQS_FULL_D(pup, 1);
872 		DEBUG_DQS_FULL_C(" window size is ", (high_limit - low_limit),
873 				 2);
874 	}
875 
876 	return MV_OK;
877 }
878 
879 /*
880  * Name:     ddr3_center_calc
881  * Desc:     Execute the calculate the center of windows phase.
882  * Args:     pDram Info
883  *           is_tx             Indicate whether Rx or Tx
884  * Notes:
885  * Returns:  MV_OK if success, other error code if fail.
886  */
887 static int ddr3_center_calc(MV_DRAM_INFO *dram_info, u32 cs, u32 ecc,
888 			    int is_tx)
889 {
890 	/* bit array of pups that need specail search */
891 	u32 special_pattern_i_pup = 0;
892 	u32 special_pattern_ii_pup = 0;
893 	u32 pup;
894 	u32 max_pup;
895 
896 	max_pup = (ecc + (1 - ecc) * dram_info->num_of_std_pups);
897 
898 	for (pup = 0; pup < max_pup; pup++) {
899 		if (is_tx == 0) {
900 			/* Check special pattern I */
901 			/*
902 			 * Special pattern Low limit search - relevant only
903 			 * for Rx, win size < threshold and low limit = 0
904 			 */
905 			if (((centralization_high_limit[pup] -
906 			      centralization_low_limit[pup]) < VALID_WIN_THRS)
907 			    && (centralization_low_limit[pup] == MIN_DELAY))
908 				special_pattern_i_pup |= (1 << pup);
909 
910 			/* Check special pattern II */
911 			/*
912 			 * Special pattern High limit search - relevant only
913 			 * for Rx, win size < threshold and high limit = 31
914 			 */
915 			if (((centralization_high_limit[pup] -
916 			      centralization_low_limit[pup]) < VALID_WIN_THRS)
917 			    && (centralization_high_limit[pup] == MAX_DELAY))
918 				special_pattern_ii_pup |= (1 << pup);
919 		}
920 	}
921 
922 	/* Run special pattern Low limit search - for relevant pup */
923 	if (special_pattern_i_pup != 0) {
924 		DEBUG_DQS_S("DDR3 - DQS Center Calc - Entering special pattern I for Low limit search\n");
925 		if (MV_OK !=
926 		    ddr3_special_pattern_i_search(dram_info, cs, ecc, is_tx,
927 					      special_pattern_i_pup))
928 			return MV_DDR3_TRAINING_ERR_DQS_LOW_LIMIT_SEARCH;
929 	}
930 
931 	/* Run special pattern High limit search - for relevant pup */
932 	if (special_pattern_ii_pup != 0) {
933 		DEBUG_DQS_S("DDR3 - DQS Center Calc - Entering special pattern II for High limit search\n");
934 		if (MV_OK !=
935 		    ddr3_special_pattern_ii_search(dram_info, cs, ecc, is_tx,
936 						   special_pattern_ii_pup))
937 			return MV_DDR3_TRAINING_ERR_DQS_HIGH_LIMIT_SEARCH;
938 	}
939 
940 	/* Set adll to center = (General_High_limit + General_Low_limit)/2 */
941 	return ddr3_set_dqs_centralization_results(dram_info, cs, ecc, is_tx);
942 }
943 
944 /*
945  * Name:     ddr3_special_pattern_i_search
946  * Desc:     Execute special pattern low limit search.
947  * Args:
948  *           special_pattern_pup  The pups that need the special search
949  * Notes:
950  * Returns:  MV_OK if success, other error code if fail.
951  */
952 int ddr3_special_pattern_i_search(MV_DRAM_INFO *dram_info, u32 cs, u32 ecc,
953 				  int is_tx, u32 special_pattern_pup)
954 {
955 	u32 victim_dq;		/* loop index - victim DQ */
956 	u32 adll_idx;
957 	u32 pup;
958 	u32 unlock_pup;		/* bit array of the unlock pups  */
959 	u32 first_fail;	/* bit array - of pups that  get first fail */
960 	u32 new_lockup_pup;	/* bit array of compare failed pups */
961 	u32 pass_pup;		/* bit array of compare pass pup */
962 	u32 sdram_offset;
963 	u32 max_pup;
964 	u32 comp_val;
965 	u32 special_res[MAX_PUP_NUM];	/* hold tmp results */
966 
967 	DEBUG_DQS_S("DDR3 - DQS - Special Pattern I Search - Starting\n");
968 
969 	max_pup = ecc + (1 - ecc) * dram_info->num_of_std_pups;
970 
971 	/* Init the temporary results to max ADLL value */
972 	for (pup = 0; pup < max_pup; pup++)
973 		special_res[pup] = ADLL_MAX;
974 
975 	/* Run special pattern for all DQ - use the same pattern */
976 	for (victim_dq = 0; victim_dq < DQ_NUM; victim_dq++) {
977 		unlock_pup = special_pattern_pup;
978 		first_fail = 0;
979 
980 		sdram_offset = cs * SDRAM_CS_SIZE + SDRAM_DQS_RX_OFFS +
981 			LEN_KILLER_PATTERN * 4 * victim_dq;
982 
983 		for (pup = 0; pup < max_pup; pup++) {
984 			/* Set adll value per PUP. adll = high limit per pup */
985 			if (IS_PUP_ACTIVE(unlock_pup, pup)) {
986 				/* only for pups that need special search */
987 				ddr3_write_pup_reg(PUP_DQS_RD, cs,
988 						   pup + (ecc * ECC_PUP), 0,
989 						   centralization_high_limit
990 						   [pup]);
991 			}
992 		}
993 
994 		adll_idx = 0;
995 		do {
996 			/*
997 			 * Perform read and compare simultaneously for all
998 			 * un-locked MC use the special pattern mask
999 			 */
1000 			new_lockup_pup = 0;
1001 
1002 			if (MV_OK !=
1003 			    ddr3_sdram_dqs_compare(dram_info, unlock_pup,
1004 						   &new_lockup_pup,
1005 						   special_pattern
1006 						   [victim_dq],
1007 						   LEN_SPECIAL_PATTERN,
1008 						   sdram_offset, 0,
1009 						   0, NULL, 1))
1010 				return MV_FAIL;
1011 
1012 			DEBUG_DQS_S("DDR3 - DQS - Special I - ADLL value is: ");
1013 			DEBUG_DQS_D(adll_idx, 2);
1014 			DEBUG_DQS_S(", UnlockPup: ");
1015 			DEBUG_DQS_D(unlock_pup, 2);
1016 			DEBUG_DQS_S(", NewLockPup: ");
1017 			DEBUG_DQS_D(new_lockup_pup, 2);
1018 			DEBUG_DQS_S("\n");
1019 
1020 			if (unlock_pup != new_lockup_pup)
1021 				DEBUG_DQS_S("DDR3 - DQS - Special I - Some Pup passed!\n");
1022 
1023 			/* Search for pups with passed compare & already fail */
1024 			pass_pup = first_fail & ~new_lockup_pup & unlock_pup;
1025 			first_fail |= new_lockup_pup;
1026 			unlock_pup &= ~pass_pup;
1027 
1028 			/* Get pass pups */
1029 			if (pass_pup != 0) {
1030 				for (pup = 0; pup < max_pup; pup++) {
1031 					if (IS_PUP_ACTIVE(pass_pup, pup) ==
1032 					    1) {
1033 						/* If pup passed and has first fail = 1 */
1034 						/* keep min value of ADLL max value - current adll */
1035 						/* (centralization_high_limit[pup] + adll_idx) = current adll !!! */
1036 						comp_val =
1037 						    (ADLL_MAX -
1038 						     (centralization_high_limit
1039 						      [pup] + adll_idx));
1040 
1041 						DEBUG_DQS_C
1042 						    ("DDR3 - DQS - Special I - Pup - ",
1043 						     pup, 1);
1044 						DEBUG_DQS_C
1045 						    (" comp_val = ",
1046 						     comp_val, 2);
1047 
1048 						if (comp_val <
1049 						    special_res[pup]) {
1050 							special_res[pup] =
1051 							    comp_val;
1052 							centralization_low_limit
1053 							    [pup] =
1054 							    (-1) *
1055 							    comp_val;
1056 
1057 							DEBUG_DQS_C
1058 							    ("DDR3 - DQS - Special I - Pup - ",
1059 							     pup, 1);
1060 							DEBUG_DQS_C
1061 							    (" Changed Low limit to ",
1062 							     centralization_low_limit
1063 							     [pup], 2);
1064 						}
1065 					}
1066 				}
1067 			}
1068 
1069 			/*
1070 			 * Did all PUP found missing window?
1071 			 * Check for each pup if adll (different for each pup)
1072 			 * reach maximum if reach max value - lock the pup
1073 			 * if not - increment (Move to right one phase - ADLL)
1074 			 * dqs RX delay
1075 			 */
1076 			adll_idx++;
1077 			for (pup = 0; pup < max_pup; pup++) {
1078 				if (IS_PUP_ACTIVE(unlock_pup, pup) == 1) {
1079 					/* Check only unlocked pups */
1080 					if ((centralization_high_limit[pup] +
1081 					     adll_idx) >= ADLL_MAX) {
1082 						/* reach maximum - lock the pup */
1083 						DEBUG_DQS_C("DDR3 - DQS - Special I - reach maximum - lock pup ",
1084 							    pup, 1);
1085 						unlock_pup &= ~(1 << pup);
1086 					} else {
1087 						/* Didn't reach maximum - increment ADLL */
1088 						ddr3_write_pup_reg(PUP_DQS_RD,
1089 								   cs,
1090 								   pup +
1091 								   (ecc *
1092 								    ECC_PUP), 0,
1093 								   (centralization_high_limit
1094 								    [pup] +
1095 								    adll_idx));
1096 					}
1097 				}
1098 			}
1099 		} while (unlock_pup != 0);
1100 	}
1101 
1102 	return MV_OK;
1103 }
1104 
1105 /*
1106  * Name:     ddr3_special_pattern_ii_search
1107  * Desc:     Execute special pattern high limit search.
1108  * Args:
1109  *           special_pattern_pup  The pups that need the special search
1110  * Notes:
1111  * Returns:  MV_OK if success, other error code if fail.
1112  */
1113 int ddr3_special_pattern_ii_search(MV_DRAM_INFO *dram_info, u32 cs, u32 ecc,
1114 				   int is_tx, u32 special_pattern_pup)
1115 {
1116 	u32 victim_dq;		/* loop index - victim DQ */
1117 	u32 adll_idx;
1118 	u32 pup;
1119 	u32 unlock_pup;		/* bit array of the unlock pups  */
1120 	u32 first_fail;	/* bit array - of pups that  get first fail */
1121 	u32 new_lockup_pup;	/* bit array of compare failed pups */
1122 	u32 pass_pup;		/* bit array of compare pass pup */
1123 	u32 sdram_offset;
1124 	u32 max_pup;
1125 	u32 comp_val;
1126 	u32 special_res[MAX_PUP_NUM];	/* hold tmp results */
1127 
1128 	DEBUG_DQS_S("DDR3 - DQS - Special Pattern II Search - Starting\n");
1129 
1130 	max_pup = (ecc + (1 - ecc) * dram_info->num_of_std_pups);
1131 
1132 	/* init the tmporary results to max ADLL value */
1133 	for (pup = 0; pup < max_pup; pup++)
1134 		special_res[pup] = ADLL_MAX;
1135 
1136 	sdram_offset = cs * SDRAM_CS_SIZE + SDRAM_DQS_RX_OFFS;
1137 
1138 	/* run special pattern for all DQ - use the same pattern */
1139 	for (victim_dq = 0; victim_dq < DQ_NUM; victim_dq++) {
1140 		unlock_pup = special_pattern_pup;
1141 		first_fail = 0;
1142 
1143 		for (pup = 0; pup < max_pup; pup++) {
1144 			/* Set adll value per PUP. adll = 0 */
1145 			if (IS_PUP_ACTIVE(unlock_pup, pup)) {
1146 				/* Only for pups that need special search */
1147 				ddr3_write_pup_reg(PUP_DQS_RD, cs,
1148 						   pup + (ecc * ECC_PUP), 0,
1149 						   ADLL_MIN);
1150 			}
1151 		}
1152 
1153 		adll_idx = 0;
1154 		do {
1155 			/*
1156 			 * Perform read and compare simultaneously for all
1157 			 * un-locked MC use the special pattern mask
1158 			 */
1159 			new_lockup_pup = 0;
1160 
1161 			if (MV_OK != ddr3_sdram_dqs_compare(
1162 				    dram_info, unlock_pup, &new_lockup_pup,
1163 				    special_pattern[victim_dq],
1164 				    LEN_SPECIAL_PATTERN,
1165 				    sdram_offset, 0, 0, NULL, 0))
1166 				return MV_FAIL;
1167 
1168 			DEBUG_DQS_S("DDR3 - DQS - Special II - ADLL value is ");
1169 			DEBUG_DQS_D(adll_idx, 2);
1170 			DEBUG_DQS_S("unlock_pup ");
1171 			DEBUG_DQS_D(unlock_pup, 1);
1172 			DEBUG_DQS_S("new_lockup_pup ");
1173 			DEBUG_DQS_D(new_lockup_pup, 1);
1174 			DEBUG_DQS_S("\n");
1175 
1176 			if (unlock_pup != new_lockup_pup) {
1177 				DEBUG_DQS_S("DDR3 - DQS - Special II - Some Pup passed!\n");
1178 			}
1179 
1180 			/* Search for pups with passed compare & already fail */
1181 			pass_pup = first_fail & ~new_lockup_pup & unlock_pup;
1182 			first_fail |= new_lockup_pup;
1183 			unlock_pup &= ~pass_pup;
1184 
1185 			/* Get pass pups */
1186 			if (pass_pup != 0) {
1187 				for (pup = 0; pup < max_pup; pup++) {
1188 					if (IS_PUP_ACTIVE(pass_pup, pup) ==
1189 					    1) {
1190 						/* If pup passed and has first fail = 1 */
1191 						/* keep min value of ADLL max value - current adll */
1192 						/* (adll_idx) = current adll !!! */
1193 						comp_val = adll_idx;
1194 
1195 						DEBUG_DQS_C("DDR3 - DQS - Special II - Pup - ",
1196 							    pup, 1);
1197 						DEBUG_DQS_C(" comp_val = ",
1198 							    comp_val, 1);
1199 
1200 						if (comp_val <
1201 						    special_res[pup]) {
1202 							special_res[pup] =
1203 							    comp_val;
1204 							centralization_high_limit
1205 							    [pup] =
1206 							    ADLL_MAX +
1207 							    comp_val;
1208 
1209 							DEBUG_DQS_C
1210 							    ("DDR3 - DQS - Special II - Pup - ",
1211 							     pup, 1);
1212 							DEBUG_DQS_C
1213 							    (" Changed High limit to ",
1214 							     centralization_high_limit
1215 							     [pup], 2);
1216 						}
1217 					}
1218 				}
1219 			}
1220 
1221 			/*
1222 			 * Did all PUP found missing window?
1223 			 * Check for each pup if adll (different for each pup)
1224 			 * reach maximum if reach max value - lock the pup
1225 			 * if not - increment (Move to right one phase - ADLL)
1226 			 * dqs RX delay
1227 			 */
1228 			adll_idx++;
1229 			for (pup = 0; pup < max_pup; pup++) {
1230 				if (IS_PUP_ACTIVE(unlock_pup, pup) == 1) {
1231 					/* Check only unlocked pups */
1232 					if ((adll_idx) >= ADLL_MAX) {
1233 						/* Reach maximum - lock the pup */
1234 						DEBUG_DQS_C("DDR3 - DQS - Special II - reach maximum - lock pup ",
1235 							    pup, 1);
1236 						unlock_pup &= ~(1 << pup);
1237 					} else {
1238 						/* Didn't reach maximum - increment ADLL */
1239 						ddr3_write_pup_reg(PUP_DQS_RD,
1240 								   cs,
1241 								   pup +
1242 								   (ecc *
1243 								    ECC_PUP), 0,
1244 								   (adll_idx));
1245 					}
1246 				}
1247 			}
1248 		} while (unlock_pup != 0);
1249 	}
1250 
1251 	return MV_OK;
1252 }
1253 
1254 /*
1255  * Name:     ddr3_set_dqs_centralization_results
1256  * Desc:     Set to HW the DQS centralization phase results.
1257  * Args:
1258  *           is_tx             Indicates whether to set Tx or RX results
1259  * Notes:
1260  * Returns:  MV_OK if success, other error code if fail.
1261  */
1262 int ddr3_set_dqs_centralization_results(MV_DRAM_INFO *dram_info, u32 cs,
1263 					u32 ecc, int is_tx)
1264 {
1265 	u32 pup, pup_num;
1266 	int addl_val;
1267 	u32 max_pup;
1268 
1269 	max_pup = (ecc + (1 - ecc) * dram_info->num_of_std_pups);
1270 
1271 	DEBUG_DQS_RESULTS_S("\n############ LOG LEVEL 2(Windows margins) ############\n");;
1272 
1273 	if (is_tx) {
1274 		DEBUG_DQS_RESULTS_C("DDR3 - DQS TX - Set Dqs Centralization Results - CS: ",
1275 				    cs, 1);
1276 	} else {
1277 		DEBUG_DQS_RESULTS_C("DDR3 - DQS RX - Set Dqs Centralization Results - CS: ",
1278 				    cs, 1);
1279 	}
1280 
1281 	/* Set adll to center = (General_High_limit + General_Low_limit)/2 */
1282 	DEBUG_DQS_RESULTS_S("\nDQS    LOW     HIGH     WIN-SIZE      Set\n");
1283 	DEBUG_DQS_RESULTS_S("==============================================\n");
1284 	for (pup = 0; pup < max_pup; pup++) {
1285 		addl_val = (centralization_high_limit[pup] +
1286 			    centralization_low_limit[pup]) / 2;
1287 
1288 		pup_num = pup * (1 - ecc) + ecc * ECC_PUP;
1289 
1290 		DEBUG_DQS_RESULTS_D(pup_num, 1);
1291 		DEBUG_DQS_RESULTS_S("     0x");
1292 		DEBUG_DQS_RESULTS_D(centralization_low_limit[pup], 2);
1293 		DEBUG_DQS_RESULTS_S("      0x");
1294 		DEBUG_DQS_RESULTS_D(centralization_high_limit[pup], 2);
1295 		DEBUG_DQS_RESULTS_S("      0x");
1296 		DEBUG_DQS_RESULTS_D(centralization_high_limit[pup] -
1297 				    centralization_low_limit[pup], 2);
1298 		DEBUG_DQS_RESULTS_S("       0x");
1299 		DEBUG_DQS_RESULTS_D(addl_val, 2);
1300 		DEBUG_DQS_RESULTS_S("\n");
1301 
1302 		if (addl_val < ADLL_MIN) {
1303 			addl_val = ADLL_MIN;
1304 			DEBUG_DQS_RESULTS_S("DDR3 - DQS - Setting ADLL value for Pup to MIN (since it was lower than 0)\n");
1305 		}
1306 
1307 		if (addl_val > ADLL_MAX) {
1308 			addl_val = ADLL_MAX;
1309 			DEBUG_DQS_RESULTS_S("DDR3 - DQS - Setting ADLL value for Pup to MAX (since it was higher than 31)\n");
1310 		}
1311 
1312 		if (is_tx) {
1313 			ddr3_write_pup_reg(PUP_DQS_WR, cs, pup_num, 0,
1314 					   addl_val +
1315 					   dram_info->wl_val[cs][pup_num][D]);
1316 		} else {
1317 			ddr3_write_pup_reg(PUP_DQS_RD, cs, pup_num, 0,
1318 					   addl_val);
1319 		}
1320 	}
1321 
1322 	return MV_OK;
1323 }
1324 
1325 /*
1326  * Set training patterns
1327  */
1328 int ddr3_load_dqs_patterns(MV_DRAM_INFO *dram_info)
1329 {
1330 	u32 cs, cs_count, cs_tmp, victim_dq;
1331 	u32 sdram_addr;
1332 	u32 *pattern_ptr;
1333 
1334 	/* Loop for each CS */
1335 	for (cs = 0; cs < MAX_CS; cs++) {
1336 		if (dram_info->cs_ena & (1 << cs)) {
1337 			cs_count = 0;
1338 			for (cs_tmp = 0; cs_tmp < cs; cs_tmp++) {
1339 				if (dram_info->cs_ena & (1 << cs_tmp))
1340 					cs_count++;
1341 			}
1342 
1343 			/* Init killer pattern */
1344 			sdram_addr = (cs_count * (SDRAM_CS_SIZE + 1) +
1345 				      SDRAM_DQS_RX_OFFS);
1346 			for (victim_dq = 0; victim_dq < DQ_NUM; victim_dq++) {
1347 				pattern_ptr = ddr3_dqs_choose_pattern(dram_info,
1348 								      victim_dq);
1349 				if (MV_OK != ddr3_sdram_dqs_compare(
1350 					    dram_info, (u32)NULL, NULL,
1351 					    pattern_ptr, LEN_KILLER_PATTERN,
1352 					    sdram_addr + LEN_KILLER_PATTERN *
1353 					    4 * victim_dq, 1, 0, NULL,
1354 					    0))
1355 					return MV_DDR3_TRAINING_ERR_DQS_PATTERN;
1356 			}
1357 
1358 			/* Init special-killer pattern */
1359 			sdram_addr = (cs_count * (SDRAM_CS_SIZE + 1) +
1360 				      SDRAM_DQS_RX_SPECIAL_OFFS);
1361 			for (victim_dq = 0; victim_dq < DQ_NUM; victim_dq++) {
1362 				if (MV_OK != ddr3_sdram_dqs_compare(
1363 					    dram_info, (u32)NULL, NULL,
1364 					    special_pattern[victim_dq],
1365 					    LEN_KILLER_PATTERN, sdram_addr +
1366 					    LEN_KILLER_PATTERN * 4 * victim_dq,
1367 					    1, 0, NULL, 0))
1368 					return MV_DDR3_TRAINING_ERR_DQS_PATTERN;
1369 			}
1370 		}
1371 	}
1372 
1373 	return MV_OK;
1374 }
1375