mirror of
https://github.com/AsahiLinux/u-boot
synced 2024-11-06 13:14:27 +00:00
83d290c56f
When U-Boot started using SPDX tags we were among the early adopters and there weren't a lot of other examples to borrow from. So we picked the area of the file that usually had a full license text and replaced it with an appropriate SPDX-License-Identifier: entry. Since then, the Linux Kernel has adopted SPDX tags and they place it as the very first line in a file (except where shebangs are used, then it's second line) and with slightly different comment styles than us. In part due to community overlap, in part due to better tag visibility and in part for other minor reasons, switch over to that style. This commit changes all instances where we have a single declared license in the tag as both the before and after are identical in tag contents. There's also a few places where I found we did not have a tag and have introduced one. Signed-off-by: Tom Rini <trini@konsulko.com>
1591 lines
43 KiB
C
1591 lines
43 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
/*
|
|
* Copyright (C) Marvell International Ltd. and its affiliates
|
|
*/
|
|
|
|
#include <common.h>
|
|
#include <i2c.h>
|
|
#include <spl.h>
|
|
#include <asm/io.h>
|
|
#include <asm/arch/cpu.h>
|
|
#include <asm/arch/soc.h>
|
|
|
|
#include "ddr3_hw_training.h"
|
|
|
|
/*
|
|
* Debug
|
|
*/
|
|
#define DEBUG_PBS_FULL_C(s, d, l) \
|
|
DEBUG_PBS_FULL_S(s); DEBUG_PBS_FULL_D(d, l); DEBUG_PBS_FULL_S("\n")
|
|
#define DEBUG_PBS_C(s, d, l) \
|
|
DEBUG_PBS_S(s); DEBUG_PBS_D(d, l); DEBUG_PBS_S("\n")
|
|
|
|
#ifdef MV_DEBUG_PBS
|
|
#define DEBUG_PBS_S(s) puts(s)
|
|
#define DEBUG_PBS_D(d, l) printf("%x", d)
|
|
#else
|
|
#define DEBUG_PBS_S(s)
|
|
#define DEBUG_PBS_D(d, l)
|
|
#endif
|
|
|
|
#ifdef MV_DEBUG_FULL_PBS
|
|
#define DEBUG_PBS_FULL_S(s) puts(s)
|
|
#define DEBUG_PBS_FULL_D(d, l) printf("%x", d)
|
|
#else
|
|
#define DEBUG_PBS_FULL_S(s)
|
|
#define DEBUG_PBS_FULL_D(d, l)
|
|
#endif
|
|
|
|
#if defined(MV88F78X60) || defined(MV88F672X)
|
|
|
|
/* Temp array for skew data storage */
|
|
static u32 skew_array[(MAX_PUP_NUM) * DQ_NUM] = { 0 };
|
|
|
|
/* PBS locked dq (per pup) */
|
|
extern u32 pbs_locked_dq[MAX_PUP_NUM][DQ_NUM];
|
|
extern u32 pbs_locked_dm[MAX_PUP_NUM];
|
|
extern u32 pbs_locked_value[MAX_PUP_NUM][DQ_NUM];
|
|
|
|
#if defined(MV88F672X)
|
|
extern u32 pbs_pattern[2][LEN_16BIT_PBS_PATTERN];
|
|
extern u32 pbs_pattern_32b[2][LEN_PBS_PATTERN];
|
|
#else
|
|
extern u32 pbs_pattern_32b[2][LEN_PBS_PATTERN];
|
|
extern u32 pbs_pattern_64b[2][LEN_PBS_PATTERN];
|
|
#endif
|
|
|
|
extern u32 pbs_dq_mapping[PUP_NUM_64BIT + 1][DQ_NUM];
|
|
|
|
static int ddr3_tx_shift_dqs_adll_step_before_fail(MV_DRAM_INFO *dram_info,
|
|
u32 cur_pup, u32 pbs_pattern_idx, u32 ecc);
|
|
static int ddr3_rx_shift_dqs_to_first_fail(MV_DRAM_INFO *dram_info, u32 cur_pup,
|
|
u32 pbs_pattern_idx, u32 ecc);
|
|
static int ddr3_pbs_per_bit(MV_DRAM_INFO *dram_info, int *start_over, int is_tx,
|
|
u32 *pcur_pup, u32 pbs_pattern_idx, u32 ecc);
|
|
static int ddr3_set_pbs_results(MV_DRAM_INFO *dram_info, int is_tx);
|
|
static void ddr3_pbs_write_pup_dqs_reg(u32 cs, u32 pup, u32 dqs_delay);
|
|
|
|
/*
|
|
* Name: ddr3_pbs_tx
|
|
* Desc: Execute the PBS TX phase.
|
|
* Args: dram_info ddr3 training information struct
|
|
* Notes:
|
|
* Returns: MV_OK if success, other error code if fail.
|
|
*/
|
|
int ddr3_pbs_tx(MV_DRAM_INFO *dram_info)
|
|
{
|
|
/* Array of Deskew results */
|
|
|
|
/*
|
|
* Array to hold the total sum of skew from all iterations
|
|
* (for average purpose)
|
|
*/
|
|
u32 skew_sum_array[MAX_PUP_NUM][DQ_NUM] = { {0} };
|
|
|
|
/*
|
|
* Array to hold the total average skew from both patterns
|
|
* (for average purpose)
|
|
*/
|
|
u32 pattern_skew_array[MAX_PUP_NUM][DQ_NUM] = { {0} };
|
|
|
|
u32 pbs_rep_time = 0; /* counts number of loop in case of fail */
|
|
/* bit array for unlock pups - used to repeat on the RX operation */
|
|
u32 cur_pup;
|
|
u32 max_pup;
|
|
u32 pbs_retry;
|
|
u32 pup, dq, pups, cur_max_pup, valid_pup, reg;
|
|
u32 pattern_idx;
|
|
u32 ecc;
|
|
/* indicates whether we need to start the loop again */
|
|
int start_over;
|
|
|
|
DEBUG_PBS_S("DDR3 - PBS TX - Starting PBS TX procedure\n");
|
|
|
|
pups = dram_info->num_of_total_pups;
|
|
max_pup = dram_info->num_of_total_pups;
|
|
|
|
/* Enable SW override */
|
|
reg = reg_read(REG_DRAM_TRAINING_2_ADDR) |
|
|
(1 << REG_DRAM_TRAINING_2_SW_OVRD_OFFS);
|
|
/* [0] = 1 - Enable SW override */
|
|
/* 0x15B8 - Training SW 2 Register */
|
|
reg_write(REG_DRAM_TRAINING_2_ADDR, reg);
|
|
DEBUG_PBS_S("DDR3 - PBS RX - SW Override Enabled\n");
|
|
|
|
reg = 1 << REG_DRAM_TRAINING_AUTO_OFFS;
|
|
reg_write(REG_DRAM_TRAINING_ADDR, reg); /* 0x15B0 - Training Register */
|
|
|
|
/* Running twice for 2 different patterns. each patterns - 3 times */
|
|
for (pattern_idx = 0; pattern_idx < COUNT_PBS_PATTERN; pattern_idx++) {
|
|
DEBUG_PBS_C("DDR3 - PBS TX - Working with pattern - ",
|
|
pattern_idx, 1);
|
|
|
|
/* Reset sum array */
|
|
for (pup = 0; pup < pups; pup++) {
|
|
for (dq = 0; dq < DQ_NUM; dq++)
|
|
skew_sum_array[pup][dq] = 0;
|
|
}
|
|
|
|
/*
|
|
* Perform PBS several of times (3 for each pattern).
|
|
* At the end, we'll use the average
|
|
*/
|
|
/* If there is ECC, do each PBS again with mux change */
|
|
for (pbs_retry = 0; pbs_retry < COUNT_PBS_REPEAT; pbs_retry++) {
|
|
for (ecc = 0; ecc < (dram_info->ecc_ena + 1); ecc++) {
|
|
|
|
/*
|
|
* This parameter stores the current PUP
|
|
* num - ecc mode dependent - 4-8 / 1 pups
|
|
*/
|
|
cur_max_pup = (1 - ecc) *
|
|
dram_info->num_of_std_pups + ecc;
|
|
|
|
if (ecc) {
|
|
/* Only 1 pup in this case */
|
|
valid_pup = 0x1;
|
|
} else if (cur_max_pup > 4) {
|
|
/* 64 bit - 8 pups */
|
|
valid_pup = 0xFF;
|
|
} else if (cur_max_pup == 4) {
|
|
/* 32 bit - 4 pups */
|
|
valid_pup = 0xF;
|
|
} else {
|
|
/* 16 bit - 2 pups */
|
|
valid_pup = 0x3;
|
|
}
|
|
|
|
/* ECC Support - Switch ECC Mux on ecc=1 */
|
|
reg = reg_read(REG_DRAM_TRAINING_2_ADDR) &
|
|
~(1 << REG_DRAM_TRAINING_2_ECC_MUX_OFFS);
|
|
reg |= (dram_info->ecc_ena * ecc <<
|
|
REG_DRAM_TRAINING_2_ECC_MUX_OFFS);
|
|
reg_write(REG_DRAM_TRAINING_2_ADDR, reg);
|
|
|
|
if (ecc)
|
|
DEBUG_PBS_S("DDR3 - PBS Tx - ECC Mux Enabled\n");
|
|
else
|
|
DEBUG_PBS_S("DDR3 - PBS Tx - ECC Mux Disabled\n");
|
|
|
|
/* Init iteration values */
|
|
/* Clear the locked DQs */
|
|
for (pup = 0; pup < cur_max_pup; pup++) {
|
|
for (dq = 0; dq < DQ_NUM; dq++) {
|
|
pbs_locked_dq[
|
|
pup + ecc *
|
|
(max_pup - 1)][dq] =
|
|
0;
|
|
}
|
|
}
|
|
|
|
pbs_rep_time = 0;
|
|
cur_pup = valid_pup;
|
|
start_over = 0;
|
|
|
|
/*
|
|
* Run loop On current Pattern and current
|
|
* pattern iteration (just to cover the false
|
|
* fail problem)
|
|
*/
|
|
do {
|
|
DEBUG_PBS_S("DDR3 - PBS Tx - Pbs Rep Loop is ");
|
|
DEBUG_PBS_D(pbs_rep_time, 1);
|
|
DEBUG_PBS_S(", for Retry No.");
|
|
DEBUG_PBS_D(pbs_retry, 1);
|
|
DEBUG_PBS_S("\n");
|
|
|
|
/* Set all PBS values to MIN (0) */
|
|
DEBUG_PBS_S("DDR3 - PBS Tx - Set all PBS values to MIN\n");
|
|
|
|
for (dq = 0; dq < DQ_NUM; dq++) {
|
|
ddr3_write_pup_reg(
|
|
PUP_PBS_TX +
|
|
pbs_dq_mapping[pup *
|
|
(1 - ecc) +
|
|
ecc * ECC_PUP]
|
|
[dq], CS0, (1 - ecc) *
|
|
PUP_BC + ecc * ECC_PUP, 0,
|
|
0);
|
|
}
|
|
|
|
/*
|
|
* Shift DQ ADLL right, One step before
|
|
* fail
|
|
*/
|
|
DEBUG_PBS_S("DDR3 - PBS Tx - ADLL shift right one phase before fail\n");
|
|
|
|
if (MV_OK != ddr3_tx_shift_dqs_adll_step_before_fail
|
|
(dram_info, cur_pup, pattern_idx,
|
|
ecc))
|
|
return MV_DDR3_TRAINING_ERR_PBS_ADLL_SHR_1PHASE;
|
|
|
|
/* PBS For each bit */
|
|
DEBUG_PBS_S("DDR3 - PBS Tx - perform PBS for each bit\n");
|
|
|
|
/*
|
|
* In this stage - start_over = 0
|
|
*/
|
|
if (MV_OK != ddr3_pbs_per_bit(
|
|
dram_info, &start_over, 1,
|
|
&cur_pup, pattern_idx, ecc))
|
|
return MV_DDR3_TRAINING_ERR_PBS_TX_PER_BIT;
|
|
|
|
} while ((start_over == 1) &&
|
|
(++pbs_rep_time < COUNT_PBS_STARTOVER));
|
|
|
|
if (pbs_rep_time == COUNT_PBS_STARTOVER &&
|
|
start_over == 1) {
|
|
DEBUG_PBS_S("DDR3 - PBS Tx - FAIL - Adll reach max value\n");
|
|
return MV_DDR3_TRAINING_ERR_PBS_TX_MAX_VAL;
|
|
}
|
|
|
|
DEBUG_PBS_FULL_C("DDR3 - PBS TX - values for iteration - ",
|
|
pbs_retry, 1);
|
|
for (pup = 0; pup < cur_max_pup; pup++) {
|
|
/*
|
|
* To minimize delay elements, inc
|
|
* from pbs value the min pbs val
|
|
*/
|
|
DEBUG_PBS_S("DDR3 - PBS - PUP");
|
|
DEBUG_PBS_D((pup + (ecc * ECC_PUP)), 1);
|
|
DEBUG_PBS_S(": ");
|
|
|
|
for (dq = 0; dq < DQ_NUM; dq++) {
|
|
/* Set skew value for all dq */
|
|
/*
|
|
* Bit# Deskew <- Bit# Deskew -
|
|
* last / first failing bit
|
|
* Deskew For all bits (per PUP)
|
|
* (minimize delay elements)
|
|
*/
|
|
DEBUG_PBS_S("DQ");
|
|
DEBUG_PBS_D(dq, 1);
|
|
DEBUG_PBS_S("-");
|
|
DEBUG_PBS_D(skew_array
|
|
[((pup) * DQ_NUM) +
|
|
dq], 2);
|
|
DEBUG_PBS_S(", ");
|
|
}
|
|
DEBUG_PBS_S("\n");
|
|
}
|
|
|
|
/*
|
|
* Collect the results we got on this trial
|
|
* of PBS
|
|
*/
|
|
for (pup = 0; pup < cur_max_pup; pup++) {
|
|
for (dq = 0; dq < DQ_NUM; dq++) {
|
|
skew_sum_array[pup + (ecc * (max_pup - 1))]
|
|
[dq] += skew_array
|
|
[((pup) * DQ_NUM) + dq];
|
|
}
|
|
}
|
|
|
|
/* ECC Support - Disable ECC MUX */
|
|
reg = reg_read(REG_DRAM_TRAINING_2_ADDR) &
|
|
~(1 << REG_DRAM_TRAINING_2_ECC_MUX_OFFS);
|
|
reg_write(REG_DRAM_TRAINING_2_ADDR, reg);
|
|
}
|
|
}
|
|
|
|
DEBUG_PBS_C("DDR3 - PBS TX - values for current pattern - ",
|
|
pattern_idx, 1);
|
|
for (pup = 0; pup < max_pup; pup++) {
|
|
/*
|
|
* To minimize delay elements, inc from pbs value the
|
|
* min pbs val
|
|
*/
|
|
DEBUG_PBS_S("DDR3 - PBS - PUP");
|
|
DEBUG_PBS_D(pup, 1);
|
|
DEBUG_PBS_S(": ");
|
|
|
|
for (dq = 0; dq < DQ_NUM; dq++) {
|
|
/* set skew value for all dq */
|
|
/* Bit# Deskew <- Bit# Deskew - last / first failing bit Deskew For all bits (per PUP) (minimize delay elements) */
|
|
DEBUG_PBS_S("DQ");
|
|
DEBUG_PBS_D(dq, 1);
|
|
DEBUG_PBS_S("-");
|
|
DEBUG_PBS_D(skew_sum_array[pup][dq] /
|
|
COUNT_PBS_REPEAT, 2);
|
|
DEBUG_PBS_S(", ");
|
|
}
|
|
DEBUG_PBS_S("\n");
|
|
}
|
|
|
|
/*
|
|
* Calculate the average skew for current pattern for each
|
|
* pup and each bit
|
|
*/
|
|
DEBUG_PBS_C("DDR3 - PBS TX - Average for pattern - ",
|
|
pattern_idx, 1);
|
|
|
|
for (pup = 0; pup < max_pup; pup++) {
|
|
/*
|
|
* FOR ECC only :: found min and max value for current
|
|
* pattern skew array
|
|
*/
|
|
/* Loop for all dqs */
|
|
for (dq = 0; dq < DQ_NUM; dq++) {
|
|
pattern_skew_array[pup][dq] +=
|
|
(skew_sum_array[pup][dq] /
|
|
COUNT_PBS_REPEAT);
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Calculate the average skew */
|
|
for (pup = 0; pup < max_pup; pup++) {
|
|
for (dq = 0; dq < DQ_NUM; dq++)
|
|
skew_array[((pup) * DQ_NUM) + dq] =
|
|
pattern_skew_array[pup][dq] / COUNT_PBS_PATTERN;
|
|
}
|
|
|
|
DEBUG_PBS_S("DDR3 - PBS TX - Average for all patterns:\n");
|
|
for (pup = 0; pup < max_pup; pup++) {
|
|
/*
|
|
* To minimize delay elements, inc from pbs value the min
|
|
* pbs val
|
|
*/
|
|
DEBUG_PBS_S("DDR3 - PBS - PUP");
|
|
DEBUG_PBS_D(pup, 1);
|
|
DEBUG_PBS_S(": ");
|
|
|
|
for (dq = 0; dq < DQ_NUM; dq++) {
|
|
/* Set skew value for all dq */
|
|
/*
|
|
* Bit# Deskew <- Bit# Deskew - last / first
|
|
* failing bit Deskew For all bits (per PUP)
|
|
* (minimize delay elements)
|
|
*/
|
|
DEBUG_PBS_S("DQ");
|
|
DEBUG_PBS_D(dq, 1);
|
|
DEBUG_PBS_S("-");
|
|
DEBUG_PBS_D(skew_array[(pup * DQ_NUM) + dq], 2);
|
|
DEBUG_PBS_S(", ");
|
|
}
|
|
DEBUG_PBS_S("\n");
|
|
}
|
|
|
|
/* Return ADLL to default value */
|
|
for (pup = 0; pup < max_pup; pup++) {
|
|
if (pup == (max_pup - 1) && dram_info->ecc_ena)
|
|
pup = ECC_PUP;
|
|
ddr3_pbs_write_pup_dqs_reg(CS0, pup, INIT_WL_DELAY);
|
|
}
|
|
|
|
/* Set averaged PBS results */
|
|
ddr3_set_pbs_results(dram_info, 1);
|
|
|
|
/* Disable SW override - Must be in a different stage */
|
|
/* [0]=0 - Enable SW override */
|
|
reg = reg_read(REG_DRAM_TRAINING_2_ADDR);
|
|
reg &= ~(1 << REG_DRAM_TRAINING_2_SW_OVRD_OFFS);
|
|
/* 0x15B8 - Training SW 2 Register */
|
|
reg_write(REG_DRAM_TRAINING_2_ADDR, reg);
|
|
|
|
reg = reg_read(REG_DRAM_TRAINING_1_ADDR) |
|
|
(1 << REG_DRAM_TRAINING_1_TRNBPOINT_OFFS);
|
|
reg_write(REG_DRAM_TRAINING_1_ADDR, reg);
|
|
|
|
DEBUG_PBS_S("DDR3 - PBS Tx - PBS TX ended successfuly\n");
|
|
|
|
return MV_OK;
|
|
}
|
|
|
|
/*
|
|
* Name: ddr3_tx_shift_dqs_adll_step_before_fail
|
|
* Desc: Execute the Tx shift DQ phase.
|
|
* Args: dram_info ddr3 training information struct
|
|
* cur_pup bit array of the function active pups.
|
|
* pbs_pattern_idx Index of PBS pattern
|
|
* Notes:
|
|
* Returns: MV_OK if success, other error code if fail.
|
|
*/
|
|
static int ddr3_tx_shift_dqs_adll_step_before_fail(MV_DRAM_INFO *dram_info,
|
|
u32 cur_pup,
|
|
u32 pbs_pattern_idx, u32 ecc)
|
|
{
|
|
u32 unlock_pup; /* bit array of unlock pups */
|
|
u32 new_lockup_pup; /* bit array of compare failed pups */
|
|
u32 adll_val = 4; /* INIT_WL_DELAY */
|
|
u32 cur_max_pup, pup;
|
|
u32 dqs_dly_set[MAX_PUP_NUM] = { 0 };
|
|
u32 *pattern_ptr;
|
|
|
|
/* Choose pattern */
|
|
switch (dram_info->ddr_width) {
|
|
#if defined(MV88F672X)
|
|
case 16:
|
|
pattern_ptr = (u32 *)&pbs_pattern[pbs_pattern_idx];
|
|
break;
|
|
#endif
|
|
case 32:
|
|
pattern_ptr = (u32 *)&pbs_pattern_32b[pbs_pattern_idx];
|
|
break;
|
|
#if defined(MV88F78X60)
|
|
case 64:
|
|
pattern_ptr = (u32 *)&pbs_pattern_64b[pbs_pattern_idx];
|
|
break;
|
|
#endif
|
|
default:
|
|
return MV_FAIL;
|
|
}
|
|
|
|
/* Set current pup number */
|
|
if (cur_pup == 0x1) /* Ecc mode */
|
|
cur_max_pup = 1;
|
|
else
|
|
cur_max_pup = dram_info->num_of_std_pups;
|
|
|
|
unlock_pup = cur_pup; /* '1' for each unlocked pup */
|
|
|
|
/* Loop on all ADLL Vaules */
|
|
do {
|
|
/* Loop until found first fail */
|
|
adll_val++;
|
|
|
|
/*
|
|
* Increment (Move to right - ADLL) DQ TX delay
|
|
* (broadcast to all Data PUPs)
|
|
*/
|
|
for (pup = 0; pup < cur_max_pup; pup++)
|
|
ddr3_pbs_write_pup_dqs_reg(CS0,
|
|
pup * (1 - ecc) +
|
|
ECC_PUP * ecc, adll_val);
|
|
|
|
/*
|
|
* Write and Read, compare results (read was already verified)
|
|
*/
|
|
/* 0 - all locked */
|
|
new_lockup_pup = 0;
|
|
|
|
if (MV_OK != ddr3_sdram_compare(dram_info, unlock_pup,
|
|
&new_lockup_pup,
|
|
pattern_ptr, LEN_PBS_PATTERN,
|
|
SDRAM_PBS_TX_OFFS, 1, 0,
|
|
NULL,
|
|
0))
|
|
return MV_FAIL;
|
|
|
|
unlock_pup &= ~new_lockup_pup;
|
|
|
|
DEBUG_PBS_FULL_S("Shift DQS by 2 steps for PUPs: ");
|
|
DEBUG_PBS_FULL_D(unlock_pup, 2);
|
|
DEBUG_PBS_FULL_C(", Set ADLL value = ", adll_val, 2);
|
|
|
|
/* If any PUP failed there is '1' to mark the PUP */
|
|
if (new_lockup_pup != 0) {
|
|
/*
|
|
* Decrement (Move Back to Left two steps - ADLL)
|
|
* DQ TX delay for current failed pups and save
|
|
*/
|
|
for (pup = 0; pup < cur_max_pup; pup++) {
|
|
if (((new_lockup_pup >> pup) & 0x1) &&
|
|
dqs_dly_set[pup] == 0)
|
|
dqs_dly_set[pup] = adll_val - 1;
|
|
}
|
|
}
|
|
} while ((unlock_pup != 0) && (adll_val != ADLL_MAX));
|
|
|
|
if (unlock_pup != 0) {
|
|
DEBUG_PBS_FULL_S("DDR3 - PBS Tx - Shift DQ - Adll value reached maximum\n");
|
|
|
|
for (pup = 0; pup < cur_max_pup; pup++) {
|
|
if (((unlock_pup >> pup) & 0x1) &&
|
|
dqs_dly_set[pup] == 0)
|
|
dqs_dly_set[pup] = adll_val - 1;
|
|
}
|
|
}
|
|
|
|
DEBUG_PBS_FULL_C("PBS TX one step before fail last pups locked Adll ",
|
|
adll_val - 2, 2);
|
|
|
|
/* Set the PUP DQS DLY Values */
|
|
for (pup = 0; pup < cur_max_pup; pup++)
|
|
ddr3_pbs_write_pup_dqs_reg(CS0, pup * (1 - ecc) + ECC_PUP * ecc,
|
|
dqs_dly_set[pup]);
|
|
|
|
/* Found one phase before fail */
|
|
return MV_OK;
|
|
}
|
|
|
|
/*
|
|
* Name: ddr3_pbs_rx
|
|
* Desc: Execute the PBS RX phase.
|
|
* Args: dram_info ddr3 training information struct
|
|
* Notes:
|
|
* Returns: MV_OK if success, other error code if fail.
|
|
*/
|
|
int ddr3_pbs_rx(MV_DRAM_INFO *dram_info)
|
|
{
|
|
/*
|
|
* Array to hold the total sum of skew from all iterations
|
|
* (for average purpose)
|
|
*/
|
|
u32 skew_sum_array[MAX_PUP_NUM][DQ_NUM] = { {0} };
|
|
|
|
/*
|
|
* Array to hold the total average skew from both patterns
|
|
* (for average purpose)
|
|
*/
|
|
u32 pattern_skew_array[MAX_PUP_NUM][DQ_NUM] = { {0} };
|
|
|
|
u32 pbs_rep_time = 0; /* counts number of loop in case of fail */
|
|
/* bit array for unlock pups - used to repeat on the RX operation */
|
|
u32 cur_pup;
|
|
u32 max_pup;
|
|
u32 pbs_retry;
|
|
u32 pup, dq, pups, cur_max_pup, valid_pup, reg;
|
|
u32 pattern_idx;
|
|
u32 ecc;
|
|
/* indicates whether we need to start the loop again */
|
|
int start_over;
|
|
int status;
|
|
|
|
DEBUG_PBS_S("DDR3 - PBS RX - Starting PBS RX procedure\n");
|
|
|
|
pups = dram_info->num_of_total_pups;
|
|
max_pup = dram_info->num_of_total_pups;
|
|
|
|
/* Enable SW override */
|
|
reg = reg_read(REG_DRAM_TRAINING_2_ADDR) |
|
|
(1 << REG_DRAM_TRAINING_2_SW_OVRD_OFFS);
|
|
/* [0] = 1 - Enable SW override */
|
|
/* 0x15B8 - Training SW 2 Register */
|
|
reg_write(REG_DRAM_TRAINING_2_ADDR, reg);
|
|
DEBUG_PBS_FULL_S("DDR3 - PBS RX - SW Override Enabled\n");
|
|
|
|
reg = 1 << REG_DRAM_TRAINING_AUTO_OFFS;
|
|
reg_write(REG_DRAM_TRAINING_ADDR, reg); /* 0x15B0 - Training Register */
|
|
|
|
/* Running twice for 2 different patterns. each patterns - 3 times */
|
|
for (pattern_idx = 0; pattern_idx < COUNT_PBS_PATTERN; pattern_idx++) {
|
|
DEBUG_PBS_FULL_C("DDR3 - PBS RX - Working with pattern - ",
|
|
pattern_idx, 1);
|
|
|
|
/* Reset sum array */
|
|
for (pup = 0; pup < pups; pup++) {
|
|
for (dq = 0; dq < DQ_NUM; dq++)
|
|
skew_sum_array[pup][dq] = 0;
|
|
}
|
|
|
|
/*
|
|
* Perform PBS several of times (3 for each pattern).
|
|
* At the end, we'll use the average
|
|
*/
|
|
/* If there is ECC, do each PBS again with mux change */
|
|
for (pbs_retry = 0; pbs_retry < COUNT_PBS_REPEAT; pbs_retry++) {
|
|
for (ecc = 0; ecc < (dram_info->ecc_ena + 1); ecc++) {
|
|
/*
|
|
* This parameter stores the current PUP
|
|
* num - ecc mode dependent - 4-8 / 1 pups
|
|
*/
|
|
cur_max_pup = (1 - ecc) *
|
|
dram_info->num_of_std_pups + ecc;
|
|
|
|
if (ecc) {
|
|
/* Only 1 pup in this case */
|
|
valid_pup = 0x1;
|
|
} else if (cur_max_pup > 4) {
|
|
/* 64 bit - 8 pups */
|
|
valid_pup = 0xFF;
|
|
} else if (cur_max_pup == 4) {
|
|
/* 32 bit - 4 pups */
|
|
valid_pup = 0xF;
|
|
} else {
|
|
/* 16 bit - 2 pups */
|
|
valid_pup = 0x3;
|
|
}
|
|
|
|
/* ECC Support - Switch ECC Mux on ecc=1 */
|
|
reg = reg_read(REG_DRAM_TRAINING_2_ADDR) &
|
|
~(1 << REG_DRAM_TRAINING_2_ECC_MUX_OFFS);
|
|
reg |= (dram_info->ecc_ena * ecc <<
|
|
REG_DRAM_TRAINING_2_ECC_MUX_OFFS);
|
|
reg_write(REG_DRAM_TRAINING_2_ADDR, reg);
|
|
|
|
if (ecc)
|
|
DEBUG_PBS_FULL_S("DDR3 - PBS Rx - ECC Mux Enabled\n");
|
|
else
|
|
DEBUG_PBS_FULL_S("DDR3 - PBS Rx - ECC Mux Disabled\n");
|
|
|
|
/* Init iteration values */
|
|
/* Clear the locked DQs */
|
|
for (pup = 0; pup < cur_max_pup; pup++) {
|
|
for (dq = 0; dq < DQ_NUM; dq++) {
|
|
pbs_locked_dq[
|
|
pup + ecc * (max_pup - 1)][dq] =
|
|
0;
|
|
}
|
|
}
|
|
|
|
pbs_rep_time = 0;
|
|
cur_pup = valid_pup;
|
|
start_over = 0;
|
|
|
|
/*
|
|
* Run loop On current Pattern and current
|
|
* pattern iteration (just to cover the false
|
|
* fail problem
|
|
*/
|
|
do {
|
|
DEBUG_PBS_FULL_S("DDR3 - PBS Rx - Pbs Rep Loop is ");
|
|
DEBUG_PBS_FULL_D(pbs_rep_time, 1);
|
|
DEBUG_PBS_FULL_S(", for Retry No.");
|
|
DEBUG_PBS_FULL_D(pbs_retry, 1);
|
|
DEBUG_PBS_FULL_S("\n");
|
|
|
|
/* Set all PBS values to MAX (31) */
|
|
for (pup = 0; pup < cur_max_pup; pup++) {
|
|
for (dq = 0; dq < DQ_NUM; dq++)
|
|
ddr3_write_pup_reg(
|
|
PUP_PBS_RX +
|
|
pbs_dq_mapping[
|
|
pup * (1 - ecc)
|
|
+ ecc * ECC_PUP]
|
|
[dq], CS0,
|
|
pup + ecc * ECC_PUP,
|
|
0, MAX_PBS);
|
|
}
|
|
|
|
/* Set all DQS PBS values to MIN (0) */
|
|
for (pup = 0; pup < cur_max_pup; pup++) {
|
|
ddr3_write_pup_reg(PUP_PBS_RX +
|
|
DQ_NUM, CS0,
|
|
pup +
|
|
ecc *
|
|
ECC_PUP, 0,
|
|
0);
|
|
}
|
|
|
|
/* Shift DQS, To first Fail */
|
|
DEBUG_PBS_FULL_S("DDR3 - PBS Rx - Shift RX DQS to first fail\n");
|
|
|
|
status = ddr3_rx_shift_dqs_to_first_fail
|
|
(dram_info, cur_pup,
|
|
pattern_idx, ecc);
|
|
if (MV_OK != status) {
|
|
DEBUG_PBS_S("DDR3 - PBS Rx - ddr3_rx_shift_dqs_to_first_fail failed.\n");
|
|
DEBUG_PBS_D(status, 8);
|
|
DEBUG_PBS_S("\nDDR3 - PBS Rx - SKIP.\n");
|
|
|
|
/* Reset read FIFO */
|
|
reg = reg_read(REG_DRAM_TRAINING_ADDR);
|
|
/* Start Auto Read Leveling procedure */
|
|
reg |= (1 << REG_DRAM_TRAINING_RL_OFFS);
|
|
/* 0x15B0 - Training Register */
|
|
reg_write(REG_DRAM_TRAINING_ADDR, reg);
|
|
|
|
reg = reg_read(REG_DRAM_TRAINING_2_ADDR);
|
|
reg |= ((1 << REG_DRAM_TRAINING_2_FIFO_RST_OFFS)
|
|
+ (1 << REG_DRAM_TRAINING_2_SW_OVRD_OFFS));
|
|
/* [0] = 1 - Enable SW override, [4] = 1 - FIFO reset */
|
|
/* 0x15B8 - Training SW 2 Register */
|
|
reg_write(REG_DRAM_TRAINING_2_ADDR, reg);
|
|
|
|
do {
|
|
reg = (reg_read(REG_DRAM_TRAINING_2_ADDR))
|
|
& (1 << REG_DRAM_TRAINING_2_FIFO_RST_OFFS);
|
|
} while (reg); /* Wait for '0' */
|
|
|
|
reg = reg_read(REG_DRAM_TRAINING_ADDR);
|
|
/* Clear Auto Read Leveling procedure */
|
|
reg &= ~(1 << REG_DRAM_TRAINING_RL_OFFS);
|
|
/* 0x15B0 - Training Register */
|
|
reg_write(REG_DRAM_TRAINING_ADDR, reg);
|
|
|
|
/* Set ADLL to 15 */
|
|
for (pup = 0; pup < max_pup;
|
|
pup++) {
|
|
ddr3_write_pup_reg
|
|
(PUP_DQS_RD, CS0,
|
|
pup +
|
|
(ecc * ECC_PUP), 0,
|
|
15);
|
|
}
|
|
|
|
/* Set all PBS values to MIN (0) */
|
|
for (pup = 0; pup < cur_max_pup;
|
|
pup++) {
|
|
for (dq = 0;
|
|
dq < DQ_NUM; dq++)
|
|
ddr3_write_pup_reg
|
|
(PUP_PBS_RX +
|
|
pbs_dq_mapping
|
|
[pup * (1 - ecc) +
|
|
ecc * ECC_PUP]
|
|
[dq], CS0,
|
|
pup + ecc * ECC_PUP,
|
|
0, MIN_PBS);
|
|
}
|
|
|
|
return MV_OK;
|
|
}
|
|
|
|
/* PBS For each bit */
|
|
DEBUG_PBS_FULL_S("DDR3 - PBS Rx - perform PBS for each bit\n");
|
|
/* in this stage - start_over = 0; */
|
|
if (MV_OK != ddr3_pbs_per_bit(
|
|
dram_info, &start_over,
|
|
0, &cur_pup,
|
|
pattern_idx, ecc)) {
|
|
DEBUG_PBS_S("DDR3 - PBS Rx - ddr3_pbs_per_bit failed.");
|
|
return MV_DDR3_TRAINING_ERR_PBS_RX_PER_BIT;
|
|
}
|
|
|
|
} while ((start_over == 1) &&
|
|
(++pbs_rep_time < COUNT_PBS_STARTOVER));
|
|
|
|
if (pbs_rep_time == COUNT_PBS_STARTOVER &&
|
|
start_over == 1) {
|
|
DEBUG_PBS_FULL_S("DDR3 - PBS Rx - FAIL - Algorithm failed doing RX PBS\n");
|
|
return MV_DDR3_TRAINING_ERR_PBS_RX_MAX_VAL;
|
|
}
|
|
|
|
/* Return DQS ADLL to default value - 15 */
|
|
/* Set all DQS PBS values to MIN (0) */
|
|
for (pup = 0; pup < cur_max_pup; pup++)
|
|
ddr3_write_pup_reg(PUP_DQS_RD, CS0,
|
|
pup + ecc * ECC_PUP,
|
|
0, INIT_RL_DELAY);
|
|
|
|
DEBUG_PBS_FULL_C("DDR3 - PBS RX - values for iteration - ",
|
|
pbs_retry, 1);
|
|
for (pup = 0; pup < cur_max_pup; pup++) {
|
|
/*
|
|
* To minimize delay elements, inc from
|
|
* pbs value the min pbs val
|
|
*/
|
|
DEBUG_PBS_FULL_S("DDR3 - PBS - PUP");
|
|
DEBUG_PBS_FULL_D((pup +
|
|
(ecc * ECC_PUP)), 1);
|
|
DEBUG_PBS_FULL_S(": ");
|
|
|
|
for (dq = 0; dq < DQ_NUM; dq++) {
|
|
/* Set skew value for all dq */
|
|
/*
|
|
* Bit# Deskew <- Bit# Deskew -
|
|
* last / first failing bit
|
|
* Deskew For all bits (per PUP)
|
|
* (minimize delay elements)
|
|
*/
|
|
DEBUG_PBS_FULL_S("DQ");
|
|
DEBUG_PBS_FULL_D(dq, 1);
|
|
DEBUG_PBS_FULL_S("-");
|
|
DEBUG_PBS_FULL_D(skew_array
|
|
[((pup) *
|
|
DQ_NUM) +
|
|
dq], 2);
|
|
DEBUG_PBS_FULL_S(", ");
|
|
}
|
|
DEBUG_PBS_FULL_S("\n");
|
|
}
|
|
|
|
/*
|
|
* Collect the results we got on this trial
|
|
* of PBS
|
|
*/
|
|
for (pup = 0; pup < cur_max_pup; pup++) {
|
|
for (dq = 0; dq < DQ_NUM; dq++) {
|
|
skew_sum_array
|
|
[pup + (ecc * (max_pup - 1))]
|
|
[dq] +=
|
|
skew_array[((pup) * DQ_NUM) + dq];
|
|
}
|
|
}
|
|
|
|
/* ECC Support - Disable ECC MUX */
|
|
reg = reg_read(REG_DRAM_TRAINING_2_ADDR) &
|
|
~(1 << REG_DRAM_TRAINING_2_ECC_MUX_OFFS);
|
|
reg_write(REG_DRAM_TRAINING_2_ADDR, reg);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Calculate the average skew for current pattern for each
|
|
* pup and each bit
|
|
*/
|
|
DEBUG_PBS_FULL_C("DDR3 - PBS RX - Average for pattern - ",
|
|
pattern_idx, 1);
|
|
for (pup = 0; pup < max_pup; pup++) {
|
|
/*
|
|
* FOR ECC only :: found min and max value for
|
|
* current pattern skew array
|
|
*/
|
|
/* Loop for all dqs */
|
|
for (dq = 0; dq < DQ_NUM; dq++) {
|
|
pattern_skew_array[pup][dq] +=
|
|
(skew_sum_array[pup][dq] /
|
|
COUNT_PBS_REPEAT);
|
|
}
|
|
}
|
|
|
|
DEBUG_PBS_C("DDR3 - PBS RX - values for current pattern - ",
|
|
pattern_idx, 1);
|
|
for (pup = 0; pup < max_pup; pup++) {
|
|
/*
|
|
* To minimize delay elements, inc from pbs value the
|
|
* min pbs val
|
|
*/
|
|
DEBUG_PBS_S("DDR3 - PBS RX - PUP");
|
|
DEBUG_PBS_D(pup, 1);
|
|
DEBUG_PBS_S(": ");
|
|
|
|
for (dq = 0; dq < DQ_NUM; dq++) {
|
|
/* Set skew value for all dq */
|
|
/*
|
|
* Bit# Deskew <- Bit# Deskew - last / first
|
|
* failing bit Deskew For all bits (per PUP)
|
|
* (minimize delay elements)
|
|
*/
|
|
DEBUG_PBS_S("DQ");
|
|
DEBUG_PBS_D(dq, 1);
|
|
DEBUG_PBS_S("-");
|
|
DEBUG_PBS_D(skew_sum_array[pup][dq] /
|
|
COUNT_PBS_REPEAT, 2);
|
|
DEBUG_PBS_S(", ");
|
|
}
|
|
DEBUG_PBS_S("\n");
|
|
}
|
|
}
|
|
|
|
/* Calculate the average skew */
|
|
for (pup = 0; pup < max_pup; pup++) {
|
|
for (dq = 0; dq < DQ_NUM; dq++)
|
|
skew_array[((pup) * DQ_NUM) + dq] =
|
|
pattern_skew_array[pup][dq] / COUNT_PBS_PATTERN;
|
|
}
|
|
|
|
DEBUG_PBS_S("DDR3 - PBS RX - Average for all patterns:\n");
|
|
for (pup = 0; pup < max_pup; pup++) {
|
|
/*
|
|
* To minimize delay elements, inc from pbs value the
|
|
* min pbs val
|
|
*/
|
|
DEBUG_PBS_S("DDR3 - PBS - PUP");
|
|
DEBUG_PBS_D(pup, 1);
|
|
DEBUG_PBS_S(": ");
|
|
|
|
for (dq = 0; dq < DQ_NUM; dq++) {
|
|
/* Set skew value for all dq */
|
|
/*
|
|
* Bit# Deskew <- Bit# Deskew - last / first
|
|
* failing bit Deskew For all bits (per PUP)
|
|
* (minimize delay elements)
|
|
*/
|
|
DEBUG_PBS_S("DQ");
|
|
DEBUG_PBS_D(dq, 1);
|
|
DEBUG_PBS_S("-");
|
|
DEBUG_PBS_D(skew_array[(pup * DQ_NUM) + dq], 2);
|
|
DEBUG_PBS_S(", ");
|
|
}
|
|
DEBUG_PBS_S("\n");
|
|
}
|
|
|
|
/* Return ADLL to default value */
|
|
ddr3_write_pup_reg(PUP_DQS_RD, CS0, PUP_BC, 0, INIT_RL_DELAY);
|
|
|
|
/* Set averaged PBS results */
|
|
ddr3_set_pbs_results(dram_info, 0);
|
|
|
|
/* Disable SW override - Must be in a different stage */
|
|
/* [0]=0 - Enable SW override */
|
|
reg = reg_read(REG_DRAM_TRAINING_2_ADDR);
|
|
reg &= ~(1 << REG_DRAM_TRAINING_2_SW_OVRD_OFFS);
|
|
/* 0x15B8 - Training SW 2 Register */
|
|
reg_write(REG_DRAM_TRAINING_2_ADDR, reg);
|
|
|
|
reg = reg_read(REG_DRAM_TRAINING_1_ADDR) |
|
|
(1 << REG_DRAM_TRAINING_1_TRNBPOINT_OFFS);
|
|
reg_write(REG_DRAM_TRAINING_1_ADDR, reg);
|
|
|
|
DEBUG_PBS_FULL_S("DDR3 - PBS RX - ended successfuly\n");
|
|
|
|
return MV_OK;
|
|
}
|
|
|
|
/*
|
|
* Name: ddr3_rx_shift_dqs_to_first_fail
|
|
* Desc: Execute the Rx shift DQ phase.
|
|
* Args: dram_info ddr3 training information struct
|
|
* cur_pup bit array of the function active pups.
|
|
* pbs_pattern_idx Index of PBS pattern
|
|
* Notes:
|
|
* Returns: MV_OK if success, other error code if fail.
|
|
*/
|
|
static int ddr3_rx_shift_dqs_to_first_fail(MV_DRAM_INFO *dram_info, u32 cur_pup,
|
|
u32 pbs_pattern_idx, u32 ecc)
|
|
{
|
|
u32 unlock_pup; /* bit array of unlock pups */
|
|
u32 new_lockup_pup; /* bit array of compare failed pups */
|
|
u32 adll_val = MAX_DELAY;
|
|
u32 dqs_deskew_val = 0; /* current value of DQS PBS deskew */
|
|
u32 cur_max_pup, pup, pass_pup;
|
|
u32 *pattern_ptr;
|
|
|
|
/* Choose pattern */
|
|
switch (dram_info->ddr_width) {
|
|
#if defined(MV88F672X)
|
|
case 16:
|
|
pattern_ptr = (u32 *)&pbs_pattern[pbs_pattern_idx];
|
|
break;
|
|
#endif
|
|
case 32:
|
|
pattern_ptr = (u32 *)&pbs_pattern_32b[pbs_pattern_idx];
|
|
break;
|
|
#if defined(MV88F78X60)
|
|
case 64:
|
|
pattern_ptr = (u32 *)&pbs_pattern_64b[pbs_pattern_idx];
|
|
break;
|
|
#endif
|
|
default:
|
|
return MV_FAIL;
|
|
}
|
|
|
|
/* Set current pup number */
|
|
if (cur_pup == 0x1) /* Ecc mode */
|
|
cur_max_pup = 1;
|
|
else
|
|
cur_max_pup = dram_info->num_of_std_pups;
|
|
|
|
unlock_pup = cur_pup; /* '1' for each unlocked pup */
|
|
|
|
DEBUG_PBS_FULL_S("DDR3 - PBS RX - Shift DQS - Starting...\n");
|
|
|
|
/* Set DQS ADLL to MAX */
|
|
DEBUG_PBS_FULL_S("DDR3 - PBS RX - Shift DQS - Set DQS ADLL to Max for all PUPs\n");
|
|
for (pup = 0; pup < cur_max_pup; pup++)
|
|
ddr3_write_pup_reg(PUP_DQS_RD, CS0, pup + ecc * ECC_PUP, 0,
|
|
MAX_DELAY);
|
|
|
|
/* Loop on all ADLL Vaules */
|
|
do {
|
|
/* Loop until found fail for all pups */
|
|
new_lockup_pup = 0;
|
|
if (MV_OK != ddr3_sdram_compare(dram_info, unlock_pup,
|
|
&new_lockup_pup,
|
|
pattern_ptr, LEN_PBS_PATTERN,
|
|
SDRAM_PBS_I_OFFS +
|
|
pbs_pattern_idx * SDRAM_PBS_NEXT_OFFS,
|
|
0, 0, NULL, 0)) {
|
|
DEBUG_PBS_S("DDR3 - PBS Rx - Shift DQS - MV_DDR3_TRAINING_ERR_PBS_SHIFT_QDS_SRAM_CMP(ddr3_sdram_compare)\n");
|
|
return MV_DDR3_TRAINING_ERR_PBS_SHIFT_QDS_SRAM_CMP;
|
|
}
|
|
|
|
if ((new_lockup_pup != 0) && (dqs_deskew_val <= 1)) {
|
|
/* Fail on start with first deskew value */
|
|
/* Decrement DQS ADLL */
|
|
--adll_val;
|
|
if (adll_val == ADLL_MIN) {
|
|
DEBUG_PBS_S("DDR3 - PBS Rx - Shift DQS - fail on start with first deskew value\n");
|
|
return MV_DDR3_TRAINING_ERR_PBS_SHIFT_QDS_SRAM_CMP;
|
|
}
|
|
ddr3_write_pup_reg(PUP_DQS_RD, CS0, pup + ecc * ECC_PUP,
|
|
0, adll_val);
|
|
continue;
|
|
}
|
|
|
|
/* Update all new locked pups */
|
|
unlock_pup &= ~new_lockup_pup;
|
|
|
|
if ((unlock_pup == 0) || (dqs_deskew_val == MAX_PBS)) {
|
|
if (dqs_deskew_val == MAX_PBS) {
|
|
/*
|
|
* Reach max value of dqs deskew or get fail
|
|
* for all pups
|
|
*/
|
|
DEBUG_PBS_FULL_S("DDR3 - PBS RX - Shift DQS - DQS deskew reached maximum value\n");
|
|
}
|
|
break;
|
|
}
|
|
|
|
DEBUG_PBS_FULL_S("DDR3 - PBS RX - Shift DQS - Inc DQS deskew for PUPs: ");
|
|
DEBUG_PBS_FULL_D(unlock_pup, 2);
|
|
DEBUG_PBS_FULL_C(", deskew = ", dqs_deskew_val, 2);
|
|
|
|
/* Increment DQS deskew elements - Only for unlocked pups */
|
|
dqs_deskew_val++;
|
|
for (pup = 0; pup < cur_max_pup; pup++) {
|
|
if (IS_PUP_ACTIVE(unlock_pup, pup) == 1) {
|
|
ddr3_write_pup_reg(PUP_PBS_RX + DQS_DQ_NUM, CS0,
|
|
pup + ecc * ECC_PUP, 0,
|
|
dqs_deskew_val);
|
|
}
|
|
}
|
|
} while (1);
|
|
|
|
DEBUG_PBS_FULL_S("DDR3 - PBS RX - Shift DQS - ADLL shift one step before fail\n");
|
|
/* Continue to ADLL shift one step before fail */
|
|
unlock_pup = cur_pup;
|
|
do {
|
|
/* Loop until pass compare for all pups */
|
|
new_lockup_pup = 0;
|
|
/* Read and compare results */
|
|
if (MV_OK != ddr3_sdram_compare(dram_info, unlock_pup, &new_lockup_pup,
|
|
pattern_ptr, LEN_PBS_PATTERN,
|
|
SDRAM_PBS_I_OFFS +
|
|
pbs_pattern_idx * SDRAM_PBS_NEXT_OFFS,
|
|
1, 0, NULL, 0)) {
|
|
DEBUG_PBS_S("DDR3 - PBS Rx - Shift DQS - MV_DDR3_TRAINING_ERR_PBS_SHIFT_QDS_SRAM_CMP(ddr3_sdram_compare)\n");
|
|
return MV_DDR3_TRAINING_ERR_PBS_SHIFT_QDS_SRAM_CMP;
|
|
}
|
|
|
|
/*
|
|
* Get mask for pup which passed so their adll will be
|
|
* changed to 2 steps before fails
|
|
*/
|
|
pass_pup = unlock_pup & ~new_lockup_pup;
|
|
|
|
DEBUG_PBS_FULL_S("Shift DQS by 2 steps for PUPs: ");
|
|
DEBUG_PBS_FULL_D(pass_pup, 2);
|
|
DEBUG_PBS_FULL_C(", Set ADLL value = ", (adll_val - 2), 2);
|
|
|
|
/* Only for pass pups */
|
|
for (pup = 0; pup < cur_max_pup; pup++) {
|
|
if (IS_PUP_ACTIVE(pass_pup, pup) == 1) {
|
|
ddr3_write_pup_reg(PUP_DQS_RD, CS0,
|
|
pup + ecc * ECC_PUP, 0,
|
|
(adll_val - 2));
|
|
}
|
|
}
|
|
|
|
/* Locked pups that compare success */
|
|
unlock_pup &= new_lockup_pup;
|
|
|
|
if (unlock_pup == 0) {
|
|
/* All pups locked */
|
|
break;
|
|
}
|
|
|
|
/* Found error */
|
|
if (adll_val == 0) {
|
|
DEBUG_PBS_FULL_S("DDR3 - PBS Rx - Shift DQS - Adll reach min value\n");
|
|
return MV_DDR3_TRAINING_ERR_PBS_SHIFT_QDS_MAX_VAL;
|
|
}
|
|
|
|
/*
|
|
* Decrement (Move Back to Left one phase - ADLL) dqs RX delay
|
|
*/
|
|
adll_val--;
|
|
for (pup = 0; pup < cur_max_pup; pup++) {
|
|
if (IS_PUP_ACTIVE(unlock_pup, pup) == 1) {
|
|
ddr3_write_pup_reg(PUP_DQS_RD, CS0,
|
|
pup + ecc * ECC_PUP, 0,
|
|
adll_val);
|
|
}
|
|
}
|
|
} while (1);
|
|
|
|
return MV_OK;
|
|
}
|
|
|
|
/*
|
|
* lock_pups() extracted from ddr3_pbs_per_bit(). This just got too
|
|
* much indented making it hard to read / edit.
|
|
*/
|
|
static void lock_pups(u32 pup, u32 *pup_locked, u8 *unlock_pup_dq_array,
|
|
u32 pbs_curr_val, u32 start_pbs, u32 ecc, int is_tx)
|
|
{
|
|
u32 dq;
|
|
int idx;
|
|
|
|
/* Lock PBS value for all remaining PUPs bits */
|
|
DEBUG_PBS_FULL_S("DDR3 - PBS Per bit - Lock PBS value for all remaining PUPs bits, pup ");
|
|
DEBUG_PBS_FULL_D(pup, 1);
|
|
DEBUG_PBS_FULL_C(" pbs value ", pbs_curr_val, 2);
|
|
|
|
idx = pup * (1 - ecc) + ecc * ECC_PUP;
|
|
*pup_locked &= ~(1 << pup);
|
|
|
|
for (dq = 0; dq < DQ_NUM; dq++) {
|
|
if (IS_PUP_ACTIVE(unlock_pup_dq_array[dq], pup) == 1) {
|
|
int offs;
|
|
|
|
/* Lock current dq */
|
|
unlock_pup_dq_array[dq] &= ~(1 << pup);
|
|
skew_array[(pup * DQ_NUM) + dq] = pbs_curr_val;
|
|
|
|
if (is_tx == 1)
|
|
offs = PUP_PBS_TX;
|
|
else
|
|
offs = PUP_PBS_RX;
|
|
|
|
ddr3_write_pup_reg(offs +
|
|
pbs_dq_mapping[idx][dq], CS0,
|
|
idx, 0, start_pbs);
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Name: ddr3_pbs_per_bit
|
|
* Desc: Execute the Per Bit Skew phase.
|
|
* Args: start_over Return whether need to start over the algorithm
|
|
* is_tx Indicate whether Rx or Tx
|
|
* pcur_pup bit array of the function active pups. return the
|
|
* pups that need to repeat on the PBS
|
|
* pbs_pattern_idx Index of PBS pattern
|
|
*
|
|
* Notes: Current implementation supports double activation of this function.
|
|
* i.e. in order to activate this function (using start_over) more than
|
|
* twice, the implementation should change.
|
|
* imlementation limitation are marked using
|
|
* ' CHIP-ONLY! - Implementation Limitation '
|
|
* Returns: MV_OK if success, other error code if fail.
|
|
*/
|
|
static int ddr3_pbs_per_bit(MV_DRAM_INFO *dram_info, int *start_over, int is_tx,
|
|
u32 *pcur_pup, u32 pbs_pattern_idx, u32 ecc)
|
|
{
|
|
/*
|
|
* Bit array to indicate if we already get fail on bit per pup & dq bit
|
|
*/
|
|
u8 unlock_pup_dq_array[DQ_NUM] = {
|
|
*pcur_pup, *pcur_pup, *pcur_pup, *pcur_pup, *pcur_pup,
|
|
*pcur_pup, *pcur_pup, *pcur_pup
|
|
};
|
|
|
|
u8 cmp_unlock_pup_dq_array[COUNT_PBS_COMP_RETRY_NUM][DQ_NUM];
|
|
u32 pup, dq;
|
|
/* value of pbs is according to RX or TX */
|
|
u32 start_pbs, last_pbs;
|
|
u32 pbs_curr_val;
|
|
/* bit array that indicates all dq of the pup locked */
|
|
u32 pup_locked;
|
|
u32 first_fail[MAX_PUP_NUM] = { 0 }; /* count first fail per pup */
|
|
/* indicates whether we get first fail per pup */
|
|
int first_failed[MAX_PUP_NUM] = { 0 };
|
|
/* bit array that indicates pup already get fail */
|
|
u32 sum_pup_fail;
|
|
/* use to calculate diff between curr pbs to first fail pbs */
|
|
u32 calc_pbs_diff;
|
|
u32 pbs_cmp_retry;
|
|
u32 max_pup;
|
|
|
|
/* Set init values for retry array - 8 retry */
|
|
for (pbs_cmp_retry = 0; pbs_cmp_retry < COUNT_PBS_COMP_RETRY_NUM;
|
|
pbs_cmp_retry++) {
|
|
for (dq = 0; dq < DQ_NUM; dq++)
|
|
cmp_unlock_pup_dq_array[pbs_cmp_retry][dq] = *pcur_pup;
|
|
}
|
|
|
|
memset(&skew_array, 0, MAX_PUP_NUM * DQ_NUM * sizeof(u32));
|
|
|
|
DEBUG_PBS_FULL_S("DDR3 - PBS Per bit - Started\n");
|
|
|
|
/* The pbs value depends if rx or tx */
|
|
if (is_tx == 1) {
|
|
start_pbs = MIN_PBS;
|
|
last_pbs = MAX_PBS;
|
|
} else {
|
|
start_pbs = MAX_PBS;
|
|
last_pbs = MIN_PBS;
|
|
}
|
|
|
|
pbs_curr_val = start_pbs;
|
|
pup_locked = *pcur_pup;
|
|
|
|
/* Set current pup number */
|
|
if (pup_locked == 0x1) /* Ecc mode */
|
|
max_pup = 1;
|
|
else
|
|
max_pup = dram_info->num_of_std_pups;
|
|
|
|
do {
|
|
/* Increment/ decrement PBS for un-lock bits only */
|
|
if (is_tx == 1)
|
|
pbs_curr_val++;
|
|
else
|
|
pbs_curr_val--;
|
|
|
|
/* Set Current PBS delay */
|
|
for (dq = 0; dq < DQ_NUM; dq++) {
|
|
/* Check DQ bits to see if locked in all pups */
|
|
if (unlock_pup_dq_array[dq] == 0) {
|
|
DEBUG_PBS_FULL_S("DDR3 - PBS Per bit - All pups are locked for DQ ");
|
|
DEBUG_PBS_FULL_D(dq, 1);
|
|
DEBUG_PBS_FULL_S("\n");
|
|
continue;
|
|
}
|
|
|
|
for (pup = 0; pup < max_pup; pup++) {
|
|
int idx;
|
|
|
|
idx = pup * (1 - ecc) + ecc * ECC_PUP;
|
|
|
|
if (IS_PUP_ACTIVE(unlock_pup_dq_array[dq], pup)
|
|
== 0)
|
|
continue;
|
|
|
|
if (is_tx == 1)
|
|
ddr3_write_pup_reg(
|
|
PUP_PBS_TX + pbs_dq_mapping[idx][dq],
|
|
CS0, idx, 0, pbs_curr_val);
|
|
else
|
|
ddr3_write_pup_reg(
|
|
PUP_PBS_RX + pbs_dq_mapping[idx][dq],
|
|
CS0, idx, 0, pbs_curr_val);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Write Read and compare results - run the test
|
|
* DDR_PBS_COMP_RETRY_NUM times
|
|
*/
|
|
/* Run number of read and write to verify */
|
|
for (pbs_cmp_retry = 0;
|
|
pbs_cmp_retry < COUNT_PBS_COMP_RETRY_NUM;
|
|
pbs_cmp_retry++) {
|
|
|
|
if (MV_OK !=
|
|
ddr3_sdram_pbs_compare(dram_info, pup_locked, is_tx,
|
|
pbs_pattern_idx,
|
|
pbs_curr_val, start_pbs,
|
|
skew_array,
|
|
cmp_unlock_pup_dq_array
|
|
[pbs_cmp_retry], ecc))
|
|
return MV_FAIL;
|
|
|
|
for (pup = 0; pup < max_pup; pup++) {
|
|
for (dq = 0; dq < DQ_NUM; dq++) {
|
|
if ((IS_PUP_ACTIVE(unlock_pup_dq_array[dq],
|
|
pup) == 1)
|
|
&& (IS_PUP_ACTIVE(cmp_unlock_pup_dq_array
|
|
[pbs_cmp_retry][dq],
|
|
pup) == 0)) {
|
|
DEBUG_PBS_FULL_S("DDR3 - PBS Per bit - PbsCurrVal: ");
|
|
DEBUG_PBS_FULL_D(pbs_curr_val, 2);
|
|
DEBUG_PBS_FULL_S(" PUP: ");
|
|
DEBUG_PBS_FULL_D(pup, 1);
|
|
DEBUG_PBS_FULL_S(" DQ: ");
|
|
DEBUG_PBS_FULL_D(dq, 1);
|
|
DEBUG_PBS_FULL_S(" - failed\n");
|
|
}
|
|
}
|
|
}
|
|
|
|
for (dq = 0; dq < DQ_NUM; dq++) {
|
|
unlock_pup_dq_array[dq] &=
|
|
cmp_unlock_pup_dq_array[pbs_cmp_retry][dq];
|
|
}
|
|
}
|
|
|
|
pup_locked = 0;
|
|
sum_pup_fail = *pcur_pup;
|
|
|
|
/* Check which DQ is failed */
|
|
for (dq = 0; dq < DQ_NUM; dq++) {
|
|
/* Summarize the locked pup */
|
|
pup_locked |= unlock_pup_dq_array[dq];
|
|
|
|
/* Check if get fail */
|
|
sum_pup_fail &= unlock_pup_dq_array[dq];
|
|
}
|
|
|
|
/* If all PUPS are locked in all DQ - Break */
|
|
if (pup_locked == 0) {
|
|
/* All pups are locked */
|
|
*start_over = 0;
|
|
DEBUG_PBS_FULL_S("DDR3 - PBS Per bit - All bit in all pups are successfully locked\n");
|
|
break;
|
|
}
|
|
|
|
/* PBS deskew elements reach max ? */
|
|
if (pbs_curr_val == last_pbs) {
|
|
DEBUG_PBS_FULL_S("DDR3 - PBS Per bit - PBS deskew elements reach max\n");
|
|
/* CHIP-ONLY! - Implementation Limitation */
|
|
*start_over = (sum_pup_fail != 0) && (!(*start_over));
|
|
*pcur_pup = pup_locked;
|
|
|
|
DEBUG_PBS_FULL_S("DDR3 - PBS Per bit - StartOver: ");
|
|
DEBUG_PBS_FULL_D(*start_over, 1);
|
|
DEBUG_PBS_FULL_S(" pup_locked: ");
|
|
DEBUG_PBS_FULL_D(pup_locked, 2);
|
|
DEBUG_PBS_FULL_S(" sum_pup_fail: ");
|
|
DEBUG_PBS_FULL_D(sum_pup_fail, 2);
|
|
DEBUG_PBS_FULL_S("\n");
|
|
|
|
/* Lock PBS value for all remaining bits */
|
|
for (pup = 0; pup < max_pup; pup++) {
|
|
/* Check if current pup already received error */
|
|
if (IS_PUP_ACTIVE(pup_locked, pup) == 1) {
|
|
/* Valid pup for current function */
|
|
if (IS_PUP_ACTIVE(sum_pup_fail, pup) ==
|
|
1 && (*start_over == 1)) {
|
|
DEBUG_PBS_FULL_C("DDR3 - PBS Per bit - skipping lock of pup (first loop of pbs)",
|
|
pup, 1);
|
|
continue;
|
|
} else
|
|
if (IS_PUP_ACTIVE(sum_pup_fail, pup)
|
|
== 1) {
|
|
DEBUG_PBS_FULL_C("DDR3 - PBS Per bit - Locking pup %d (even though it wasn't supposed to be locked)",
|
|
pup, 1);
|
|
}
|
|
|
|
/* Already got fail on the PUP */
|
|
/* Lock PBS value for all remaining bits */
|
|
DEBUG_PBS_FULL_S("DDR3 - PBS Per bit - Locking remaning DQs for pup - ");
|
|
DEBUG_PBS_FULL_D(pup, 1);
|
|
DEBUG_PBS_FULL_S(": ");
|
|
|
|
for (dq = 0; dq < DQ_NUM; dq++) {
|
|
if (IS_PUP_ACTIVE
|
|
(unlock_pup_dq_array[dq],
|
|
pup) == 1) {
|
|
DEBUG_PBS_FULL_D(dq, 1);
|
|
DEBUG_PBS_FULL_S(",");
|
|
/* set current PBS */
|
|
skew_array[((pup) *
|
|
DQ_NUM) +
|
|
dq] =
|
|
pbs_curr_val;
|
|
}
|
|
}
|
|
|
|
if (*start_over == 1) {
|
|
/*
|
|
* Reset this pup bit - when
|
|
* restart the PBS, ignore this
|
|
* pup
|
|
*/
|
|
*pcur_pup &= ~(1 << pup);
|
|
}
|
|
DEBUG_PBS_FULL_S("\n");
|
|
} else {
|
|
DEBUG_PBS_FULL_S("DDR3 - PBS Per bit - Pup ");
|
|
DEBUG_PBS_FULL_D(pup, 1);
|
|
DEBUG_PBS_FULL_C(" is not set in puplocked - ",
|
|
pup_locked, 1);
|
|
}
|
|
}
|
|
|
|
/* Need to start the PBS again */
|
|
if (*start_over == 1) {
|
|
DEBUG_PBS_FULL_S("DDR3 - PBS Per bit - false fail - returning to start\n");
|
|
return MV_OK;
|
|
}
|
|
break;
|
|
}
|
|
|
|
/* Diff Check */
|
|
for (pup = 0; pup < max_pup; pup++) {
|
|
if (IS_PUP_ACTIVE(pup_locked, pup) == 1) {
|
|
/* pup is not locked */
|
|
if (first_failed[pup] == 0) {
|
|
/* No first fail until now */
|
|
if (IS_PUP_ACTIVE(sum_pup_fail, pup) ==
|
|
0) {
|
|
/* Get first fail */
|
|
DEBUG_PBS_FULL_C("DDR3 - PBS Per bit - First fail in pup ",
|
|
pup, 1);
|
|
first_failed[pup] = 1;
|
|
first_fail[pup] = pbs_curr_val;
|
|
}
|
|
} else {
|
|
/* Already got first fail */
|
|
if (is_tx == 1) {
|
|
/* TX - inc pbs */
|
|
calc_pbs_diff = pbs_curr_val -
|
|
first_fail[pup];
|
|
} else {
|
|
/* RX - dec pbs */
|
|
calc_pbs_diff = first_fail[pup] -
|
|
pbs_curr_val;
|
|
}
|
|
|
|
if (calc_pbs_diff >= PBS_DIFF_LIMIT) {
|
|
lock_pups(pup, &pup_locked,
|
|
unlock_pup_dq_array,
|
|
pbs_curr_val,
|
|
start_pbs, ecc, is_tx);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
} while (1);
|
|
|
|
return MV_OK;
|
|
}
|
|
|
|
/*
|
|
* Name: ddr3_set_pbs_results
|
|
* Desc: Set to HW the PBS phase results.
|
|
* Args: is_tx Indicates whether to set Tx or RX results
|
|
* Notes:
|
|
* Returns: MV_OK if success, other error code if fail.
|
|
*/
|
|
static int ddr3_set_pbs_results(MV_DRAM_INFO *dram_info, int is_tx)
|
|
{
|
|
u32 pup, phys_pup, dq;
|
|
u32 max_pup; /* number of valid pups */
|
|
u32 pbs_min; /* minimal pbs val per pup */
|
|
u32 pbs_max; /* maximum pbs val per pup */
|
|
u32 val[9];
|
|
|
|
max_pup = dram_info->num_of_total_pups;
|
|
DEBUG_PBS_FULL_S("DDR3 - PBS - ddr3_set_pbs_results:\n");
|
|
|
|
/* Loop for all dqs & pups */
|
|
for (pup = 0; pup < max_pup; pup++) {
|
|
if (pup == (max_pup - 1) && dram_info->ecc_ena)
|
|
phys_pup = ECC_PUP;
|
|
else
|
|
phys_pup = pup;
|
|
|
|
/*
|
|
* To minimize delay elements, inc from pbs value the min
|
|
* pbs val
|
|
*/
|
|
pbs_min = MAX_PBS;
|
|
pbs_max = 0;
|
|
for (dq = 0; dq < DQ_NUM; dq++) {
|
|
if (pbs_min > skew_array[(pup * DQ_NUM) + dq])
|
|
pbs_min = skew_array[(pup * DQ_NUM) + dq];
|
|
|
|
if (pbs_max < skew_array[(pup * DQ_NUM) + dq])
|
|
pbs_max = skew_array[(pup * DQ_NUM) + dq];
|
|
}
|
|
|
|
pbs_max -= pbs_min;
|
|
|
|
DEBUG_PBS_FULL_S("DDR3 - PBS - PUP");
|
|
DEBUG_PBS_FULL_D(phys_pup, 1);
|
|
DEBUG_PBS_FULL_S(": Min Val = ");
|
|
DEBUG_PBS_FULL_D(pbs_min, 2);
|
|
DEBUG_PBS_FULL_C(", Max Val = ", pbs_max, 2);
|
|
|
|
val[pup] = 0;
|
|
|
|
for (dq = 0; dq < DQ_NUM; dq++) {
|
|
int idx;
|
|
int offs;
|
|
|
|
/* Set skew value for all dq */
|
|
/*
|
|
* Bit# Deskew <- Bit# Deskew - last / first
|
|
* failing bit Deskew For all bits (per PUP)
|
|
* (minimize delay elements)
|
|
*/
|
|
|
|
DEBUG_PBS_FULL_S("DQ");
|
|
DEBUG_PBS_FULL_D(dq, 1);
|
|
DEBUG_PBS_FULL_S("-");
|
|
DEBUG_PBS_FULL_D((skew_array[(pup * DQ_NUM) + dq] -
|
|
pbs_min), 2);
|
|
DEBUG_PBS_FULL_S(", ");
|
|
|
|
idx = (pup * DQ_NUM) + dq;
|
|
|
|
if (is_tx == 1)
|
|
offs = PUP_PBS_TX;
|
|
else
|
|
offs = PUP_PBS_RX;
|
|
|
|
ddr3_write_pup_reg(offs + pbs_dq_mapping[phys_pup][dq],
|
|
CS0, phys_pup, 0,
|
|
skew_array[idx] - pbs_min);
|
|
|
|
if (is_tx == 1)
|
|
val[pup] += skew_array[idx] - pbs_min;
|
|
}
|
|
|
|
DEBUG_PBS_FULL_S("\n");
|
|
|
|
/* Set the DQS the half of the Max PBS of the DQs */
|
|
if (is_tx == 1) {
|
|
ddr3_write_pup_reg(PUP_PBS_TX + 8, CS0, phys_pup, 0,
|
|
pbs_max / 2);
|
|
ddr3_write_pup_reg(PUP_PBS_TX + 0xa, CS0, phys_pup, 0,
|
|
val[pup] / 8);
|
|
} else
|
|
ddr3_write_pup_reg(PUP_PBS_RX + 8, CS0, phys_pup, 0,
|
|
pbs_max / 2);
|
|
}
|
|
|
|
return MV_OK;
|
|
}
|
|
|
|
static void ddr3_pbs_write_pup_dqs_reg(u32 cs, u32 pup, u32 dqs_delay)
|
|
{
|
|
u32 reg, delay;
|
|
|
|
reg = (ddr3_read_pup_reg(PUP_WL_MODE, cs, pup) & 0x3FF);
|
|
delay = reg & PUP_DELAY_MASK;
|
|
reg |= ((dqs_delay + delay) << REG_PHY_DQS_REF_DLY_OFFS);
|
|
reg |= REG_PHY_REGISTRY_FILE_ACCESS_OP_WR;
|
|
reg |= (pup << REG_PHY_PUP_OFFS);
|
|
reg |= ((0x4 * cs + PUP_WL_MODE) << REG_PHY_CS_OFFS);
|
|
|
|
reg_write(REG_PHY_REGISTRY_FILE_ACCESS_ADDR, reg); /* 0x16A0 */
|
|
do {
|
|
reg = reg_read(REG_PHY_REGISTRY_FILE_ACCESS_ADDR) &
|
|
REG_PHY_REGISTRY_FILE_ACCESS_OP_DONE;
|
|
} while (reg); /* Wait for '0' to mark the end of the transaction */
|
|
|
|
udelay(10);
|
|
}
|
|
|
|
/*
|
|
* Set training patterns
|
|
*/
|
|
int ddr3_load_pbs_patterns(MV_DRAM_INFO *dram_info)
|
|
{
|
|
u32 cs, cs_count, cs_tmp;
|
|
u32 sdram_addr;
|
|
u32 *pattern_ptr0, *pattern_ptr1;
|
|
|
|
/* Choose pattern */
|
|
switch (dram_info->ddr_width) {
|
|
#if defined(MV88F672X)
|
|
case 16:
|
|
pattern_ptr0 = (u32 *)&pbs_pattern[0];
|
|
pattern_ptr1 = (u32 *)&pbs_pattern[1];
|
|
break;
|
|
#endif
|
|
case 32:
|
|
pattern_ptr0 = (u32 *)&pbs_pattern_32b[0];
|
|
pattern_ptr1 = (u32 *)&pbs_pattern_32b[1];
|
|
break;
|
|
#if defined(MV88F78X60)
|
|
case 64:
|
|
pattern_ptr0 = (u32 *)&pbs_pattern_64b[0];
|
|
pattern_ptr1 = (u32 *)&pbs_pattern_64b[1];
|
|
break;
|
|
#endif
|
|
default:
|
|
return MV_FAIL;
|
|
}
|
|
|
|
/* Loop for each CS */
|
|
for (cs = 0; cs < MAX_CS; cs++) {
|
|
if (dram_info->cs_ena & (1 << cs)) {
|
|
cs_count = 0;
|
|
for (cs_tmp = 0; cs_tmp < cs; cs_tmp++) {
|
|
if (dram_info->cs_ena & (1 << cs_tmp))
|
|
cs_count++;
|
|
}
|
|
|
|
/* Init PBS I pattern */
|
|
sdram_addr = (cs_count * (SDRAM_CS_SIZE + 1) +
|
|
SDRAM_PBS_I_OFFS);
|
|
if (MV_OK !=
|
|
ddr3_sdram_compare(dram_info, (u32) NULL, NULL,
|
|
pattern_ptr0, LEN_STD_PATTERN,
|
|
sdram_addr, 1, 0, NULL,
|
|
0))
|
|
return MV_FAIL;
|
|
|
|
/* Init PBS II pattern */
|
|
sdram_addr = (cs_count * (SDRAM_CS_SIZE + 1) +
|
|
SDRAM_PBS_II_OFFS);
|
|
if (MV_OK !=
|
|
ddr3_sdram_compare(dram_info, (u32) NULL, NULL,
|
|
pattern_ptr1, LEN_STD_PATTERN,
|
|
sdram_addr, 1, 0, NULL,
|
|
0))
|
|
return MV_FAIL;
|
|
}
|
|
}
|
|
|
|
return MV_OK;
|
|
}
|
|
#endif
|