upstream u-boot with additional patches for our devices/boards:
https://lists.denx.de/pipermail/u-boot/2017-March/282789.html (AXP crashes) ;
Gbit ethernet patch for some LIME2 revisions ;
with SPI flash support
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1374 lines
38 KiB
1374 lines
38 KiB
/*
|
|
* Copyright (C) Marvell International Ltd. and its affiliates
|
|
*
|
|
* SPDX-License-Identifier: GPL-2.0
|
|
*/
|
|
|
|
#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_DQS_C(s, d, l) \
|
|
DEBUG_DQS_S(s); DEBUG_DQS_D(d, l); DEBUG_DQS_S("\n")
|
|
#define DEBUG_DQS_FULL_C(s, d, l) \
|
|
DEBUG_DQS_FULL_S(s); DEBUG_DQS_FULL_D(d, l); DEBUG_DQS_FULL_S("\n")
|
|
#define DEBUG_DQS_RESULTS_C(s, d, l) \
|
|
DEBUG_DQS_RESULTS_S(s); DEBUG_DQS_RESULTS_D(d, l); DEBUG_DQS_RESULTS_S("\n")
|
|
#define DEBUG_PER_DQ_C(s, d, l) \
|
|
puts(s); printf("%x", d); puts("\n")
|
|
|
|
#define DEBUG_DQS_RESULTS_S(s) \
|
|
debug_cond(ddr3_get_log_level() >= MV_LOG_LEVEL_2, "%s", s)
|
|
#define DEBUG_DQS_RESULTS_D(d, l) \
|
|
debug_cond(ddr3_get_log_level() >= MV_LOG_LEVEL_2, "%x", d)
|
|
|
|
#define DEBUG_PER_DQ_S(s) \
|
|
debug_cond(ddr3_get_log_level() >= MV_LOG_LEVEL_3, "%s", s)
|
|
#define DEBUG_PER_DQ_D(d, l) \
|
|
debug_cond(ddr3_get_log_level() >= MV_LOG_LEVEL_3, "%x", d)
|
|
#define DEBUG_PER_DQ_DD(d, l) \
|
|
debug_cond(ddr3_get_log_level() >= MV_LOG_LEVEL_3, "%d", d)
|
|
|
|
#ifdef MV_DEBUG_DQS
|
|
#define DEBUG_DQS_S(s) puts(s)
|
|
#define DEBUG_DQS_D(d, l) printf("%x", d)
|
|
#else
|
|
#define DEBUG_DQS_S(s)
|
|
#define DEBUG_DQS_D(d, l)
|
|
#endif
|
|
|
|
#ifdef MV_DEBUG_DQS_FULL
|
|
#define DEBUG_DQS_FULL_S(s) puts(s)
|
|
#define DEBUG_DQS_FULL_D(d, l) printf("%x", d)
|
|
#else
|
|
#define DEBUG_DQS_FULL_S(s)
|
|
#define DEBUG_DQS_FULL_D(d, l)
|
|
#endif
|
|
|
|
/* State machine for centralization - find low & high limit */
|
|
enum {
|
|
PUP_ADLL_LIMITS_STATE_FAIL,
|
|
PUP_ADLL_LIMITS_STATE_PASS,
|
|
PUP_ADLL_LIMITS_STATE_FAIL_AFTER_PASS,
|
|
};
|
|
|
|
/* Hold centralization low results */
|
|
static int centralization_low_limit[MAX_PUP_NUM] = { 0 };
|
|
/* Hold centralization high results */
|
|
static int centralization_high_limit[MAX_PUP_NUM] = { 0 };
|
|
|
|
int ddr3_find_adll_limits(MV_DRAM_INFO *dram_info, u32 cs, u32 ecc, int is_tx);
|
|
int ddr3_check_window_limits(u32 pup, int high_limit, int low_limit, int is_tx,
|
|
int *size_valid);
|
|
static int ddr3_center_calc(MV_DRAM_INFO *dram_info, u32 cs, u32 ecc,
|
|
int is_tx);
|
|
int ddr3_special_pattern_i_search(MV_DRAM_INFO *dram_info, u32 cs, u32 ecc,
|
|
int is_tx, u32 special_pattern_pup);
|
|
int ddr3_special_pattern_ii_search(MV_DRAM_INFO *dram_info, u32 cs, u32 ecc,
|
|
int is_tx, u32 special_pattern_pup);
|
|
int ddr3_set_dqs_centralization_results(MV_DRAM_INFO *dram_info, u32 cs, u32 ecc,
|
|
int is_tx);
|
|
|
|
#ifdef MV88F78X60
|
|
extern u32 killer_pattern_32b[DQ_NUM][LEN_SPECIAL_PATTERN];
|
|
extern u32 killer_pattern_64b[DQ_NUM][LEN_SPECIAL_PATTERN];
|
|
extern int per_bit_data[MAX_PUP_NUM][DQ_NUM];
|
|
#else
|
|
extern u32 killer_pattern[DQ_NUM][LEN_16BIT_KILLER_PATTERN];
|
|
extern u32 killer_pattern_32b[DQ_NUM][LEN_SPECIAL_PATTERN];
|
|
#if defined(MV88F672X)
|
|
extern int per_bit_data[MAX_PUP_NUM][DQ_NUM];
|
|
#endif
|
|
#endif
|
|
extern u32 special_pattern[DQ_NUM][LEN_SPECIAL_PATTERN];
|
|
|
|
static u32 *ddr3_dqs_choose_pattern(MV_DRAM_INFO *dram_info, u32 victim_dq)
|
|
{
|
|
u32 *pattern_ptr;
|
|
|
|
/* Choose pattern */
|
|
switch (dram_info->ddr_width) {
|
|
#if defined(MV88F672X)
|
|
case 16:
|
|
pattern_ptr = (u32 *)&killer_pattern[victim_dq];
|
|
break;
|
|
#endif
|
|
case 32:
|
|
pattern_ptr = (u32 *)&killer_pattern_32b[victim_dq];
|
|
break;
|
|
#if defined(MV88F78X60)
|
|
case 64:
|
|
pattern_ptr = (u32 *)&killer_pattern_64b[victim_dq];
|
|
break;
|
|
#endif
|
|
default:
|
|
#if defined(MV88F78X60)
|
|
pattern_ptr = (u32 *)&killer_pattern_32b[victim_dq];
|
|
#else
|
|
pattern_ptr = (u32 *)&killer_pattern[victim_dq];
|
|
#endif
|
|
break;
|
|
}
|
|
|
|
return pattern_ptr;
|
|
}
|
|
|
|
/*
|
|
* Name: ddr3_dqs_centralization_rx
|
|
* Desc: Execute the DQS centralization RX phase.
|
|
* Args: dram_info
|
|
* Notes:
|
|
* Returns: MV_OK if success, other error code if fail.
|
|
*/
|
|
int ddr3_dqs_centralization_rx(MV_DRAM_INFO *dram_info)
|
|
{
|
|
u32 cs, ecc, reg;
|
|
int status;
|
|
|
|
DEBUG_DQS_S("DDR3 - DQS Centralization RX - Starting procedure\n");
|
|
|
|
/* 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_DQS_S("DDR3 - DQS Centralization RX - SW Override Enabled\n");
|
|
|
|
reg = (1 << REG_DRAM_TRAINING_AUTO_OFFS);
|
|
reg_write(REG_DRAM_TRAINING_ADDR, reg); /* 0x15B0 - Training Register */
|
|
|
|
/* Loop for each CS */
|
|
for (cs = 0; cs < MAX_CS; cs++) {
|
|
if (dram_info->cs_ena & (1 << cs)) {
|
|
DEBUG_DQS_FULL_C("DDR3 - DQS Centralization RX - CS - ",
|
|
(u32) cs, 1);
|
|
|
|
for (ecc = 0; ecc < (dram_info->ecc_ena + 1); ecc++) {
|
|
|
|
/* 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_DQS_FULL_S("DDR3 - DQS Centralization RX - ECC Mux Enabled\n");
|
|
else
|
|
DEBUG_DQS_FULL_S("DDR3 - DQS Centralization RX - ECC Mux Disabled\n");
|
|
|
|
DEBUG_DQS_FULL_S("DDR3 - DQS Centralization RX - Find all limits\n");
|
|
|
|
status = ddr3_find_adll_limits(dram_info, cs,
|
|
ecc, 0);
|
|
if (MV_OK != status)
|
|
return status;
|
|
|
|
DEBUG_DQS_FULL_S("DDR3 - DQS Centralization RX - Start calculating center\n");
|
|
|
|
status = ddr3_center_calc(dram_info, cs, ecc,
|
|
0);
|
|
if (MV_OK != status)
|
|
return status;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* 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);
|
|
|
|
/* 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);
|
|
|
|
return MV_OK;
|
|
}
|
|
|
|
/*
|
|
* Name: ddr3_dqs_centralization_tx
|
|
* Desc: Execute the DQS centralization TX phase.
|
|
* Args: dram_info
|
|
* Notes:
|
|
* Returns: MV_OK if success, other error code if fail.
|
|
*/
|
|
int ddr3_dqs_centralization_tx(MV_DRAM_INFO *dram_info)
|
|
{
|
|
u32 cs, ecc, reg;
|
|
int status;
|
|
|
|
DEBUG_DQS_S("DDR3 - DQS Centralization TX - Starting procedure\n");
|
|
|
|
/* 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_DQS_S("DDR3 - DQS Centralization TX - SW Override Enabled\n");
|
|
|
|
reg = (1 << REG_DRAM_TRAINING_AUTO_OFFS);
|
|
reg_write(REG_DRAM_TRAINING_ADDR, reg); /* 0x15B0 - Training Register */
|
|
|
|
/* Loop for each CS */
|
|
for (cs = 0; cs < MAX_CS; cs++) {
|
|
if (dram_info->cs_ena & (1 << cs)) {
|
|
DEBUG_DQS_FULL_C("DDR3 - DQS Centralization TX - CS - ",
|
|
(u32) cs, 1);
|
|
for (ecc = 0; ecc < (dram_info->ecc_ena + 1); ecc++) {
|
|
/* 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_DQS_FULL_S("DDR3 - DQS Centralization TX - ECC Mux Enabled\n");
|
|
else
|
|
DEBUG_DQS_FULL_S("DDR3 - DQS Centralization TX - ECC Mux Disabled\n");
|
|
|
|
DEBUG_DQS_FULL_S("DDR3 - DQS Centralization TX - Find all limits\n");
|
|
|
|
status = ddr3_find_adll_limits(dram_info, cs,
|
|
ecc, 1);
|
|
if (MV_OK != status)
|
|
return status;
|
|
|
|
DEBUG_DQS_FULL_S("DDR3 - DQS Centralization TX - Start calculating center\n");
|
|
|
|
status = ddr3_center_calc(dram_info, cs, ecc,
|
|
1);
|
|
if (MV_OK != status)
|
|
return status;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* 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);
|
|
|
|
/* 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);
|
|
|
|
return MV_OK;
|
|
}
|
|
|
|
/*
|
|
* Name: ddr3_find_adll_limits
|
|
* Desc: Execute the Find ADLL limits phase.
|
|
* Args: dram_info
|
|
* cs
|
|
* ecc_ena
|
|
* is_tx Indicate whether Rx or Tx
|
|
* Notes:
|
|
* Returns: MV_OK if success, other error code if fail.
|
|
*/
|
|
int ddr3_find_adll_limits(MV_DRAM_INFO *dram_info, u32 cs, u32 ecc, int is_tx)
|
|
{
|
|
u32 victim_dq, pup, tmp;
|
|
u32 adll_addr;
|
|
u32 max_pup; /* maximal pup index */
|
|
u32 pup_mask = 0;
|
|
u32 unlock_pup; /* bit array of un locked pups */
|
|
u32 new_unlock_pup; /* bit array of compare failed pups */
|
|
u32 curr_adll;
|
|
u32 adll_start_val; /* adll start loop value - for rx or tx limit */
|
|
u32 high_limit; /* holds found High Limit */
|
|
u32 low_limit; /* holds found Low Limit */
|
|
int win_valid;
|
|
int update_win;
|
|
u32 sdram_offset;
|
|
u32 uj, cs_count, cs_tmp, ii;
|
|
u32 *pattern_ptr;
|
|
u32 dq;
|
|
u32 adll_end_val; /* adll end of loop val - for rx or tx limit */
|
|
u8 analog_pbs[DQ_NUM][MAX_PUP_NUM][DQ_NUM][2];
|
|
u8 analog_pbs_sum[MAX_PUP_NUM][DQ_NUM][2];
|
|
int pup_adll_limit_state[MAX_PUP_NUM]; /* hold state of each pup */
|
|
|
|
adll_addr = ((is_tx == 1) ? PUP_DQS_WR : PUP_DQS_RD);
|
|
adll_end_val = ((is_tx == 1) ? ADLL_MIN : ADLL_MAX);
|
|
adll_start_val = ((is_tx == 1) ? ADLL_MAX : ADLL_MIN);
|
|
max_pup = (ecc + (1 - ecc) * dram_info->num_of_std_pups);
|
|
|
|
DEBUG_DQS_FULL_S("DDR3 - DQS Find Limits - Starting Find ADLL Limits\n");
|
|
|
|
/* init the array */
|
|
for (pup = 0; pup < max_pup; pup++) {
|
|
centralization_low_limit[pup] = ADLL_MIN;
|
|
centralization_high_limit[pup] = ADLL_MAX;
|
|
}
|
|
|
|
/* Killer Pattern */
|
|
cs_count = 0;
|
|
for (cs_tmp = 0; cs_tmp < cs; cs_tmp++) {
|
|
if (dram_info->cs_ena & (1 << cs_tmp))
|
|
cs_count++;
|
|
}
|
|
sdram_offset = cs_count * (SDRAM_CS_SIZE + 1);
|
|
sdram_offset += ((is_tx == 1) ?
|
|
SDRAM_DQS_TX_OFFS : SDRAM_DQS_RX_OFFS);
|
|
|
|
/* Prepare pup masks */
|
|
for (pup = 0; pup < max_pup; pup++)
|
|
pup_mask |= (1 << pup);
|
|
|
|
for (pup = 0; pup < max_pup; pup++) {
|
|
for (dq = 0; dq < DQ_NUM; dq++) {
|
|
analog_pbs_sum[pup][dq][0] = adll_start_val;
|
|
analog_pbs_sum[pup][dq][1] = adll_end_val;
|
|
}
|
|
}
|
|
|
|
/* Loop - use different pattern for each victim_dq */
|
|
for (victim_dq = 0; victim_dq < DQ_NUM; victim_dq++) {
|
|
DEBUG_DQS_FULL_C("DDR3 - DQS Find Limits - Victim DQ - ",
|
|
(u32)victim_dq, 1);
|
|
/*
|
|
* The pups 3 bit arrays represent state machine. with
|
|
* 3 stages for each pup.
|
|
* 1. fail and didn't get pass in earlier compares.
|
|
* 2. pass compare
|
|
* 3. fail after pass - end state.
|
|
* The window limits are the adll values where the adll
|
|
* was in the pass stage.
|
|
*/
|
|
|
|
/* Set all states to Fail (1st state) */
|
|
for (pup = 0; pup < max_pup; pup++)
|
|
pup_adll_limit_state[pup] = PUP_ADLL_LIMITS_STATE_FAIL;
|
|
|
|
/* Set current valid pups */
|
|
unlock_pup = pup_mask;
|
|
|
|
/* Set ADLL to start value */
|
|
curr_adll = adll_start_val;
|
|
|
|
#if defined(MV88F78X60)
|
|
for (pup = 0; pup < max_pup; pup++) {
|
|
for (dq = 0; dq < DQ_NUM; dq++) {
|
|
analog_pbs[victim_dq][pup][dq][0] =
|
|
adll_start_val;
|
|
analog_pbs[victim_dq][pup][dq][1] =
|
|
adll_end_val;
|
|
per_bit_data[pup][dq] = 0;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
for (uj = 0; uj < ADLL_MAX; uj++) {
|
|
DEBUG_DQS_FULL_C("DDR3 - DQS Find Limits - Setting ADLL to ",
|
|
curr_adll, 2);
|
|
for (pup = 0; pup < max_pup; pup++) {
|
|
if (IS_PUP_ACTIVE(unlock_pup, pup) == 1) {
|
|
tmp = ((is_tx == 1) ? curr_adll +
|
|
dram_info->wl_val[cs]
|
|
[pup * (1 - ecc) + ecc * ECC_PUP]
|
|
[D] : curr_adll);
|
|
ddr3_write_pup_reg(adll_addr, cs, pup +
|
|
(ecc * ECC_PUP), 0, tmp);
|
|
}
|
|
}
|
|
|
|
/* Choose pattern */
|
|
pattern_ptr = ddr3_dqs_choose_pattern(dram_info,
|
|
victim_dq);
|
|
|
|
/* '1' - means pup failed, '0' - means pup pass */
|
|
new_unlock_pup = 0;
|
|
|
|
/* Read and compare results for Victim_DQ# */
|
|
for (ii = 0; ii < 3; ii++) {
|
|
u32 tmp = 0;
|
|
if (MV_OK != ddr3_sdram_dqs_compare(dram_info,
|
|
unlock_pup, &tmp,
|
|
pattern_ptr,
|
|
LEN_KILLER_PATTERN,
|
|
sdram_offset +
|
|
LEN_KILLER_PATTERN *
|
|
4 * victim_dq,
|
|
is_tx, 0, NULL,
|
|
0))
|
|
return MV_DDR3_TRAINING_ERR_DRAM_COMPARE;
|
|
|
|
new_unlock_pup |= tmp;
|
|
}
|
|
|
|
pup = 0;
|
|
DEBUG_DQS_FULL_C("DDR3 - DQS Find Limits - UnlockPup: ",
|
|
unlock_pup, 2);
|
|
DEBUG_DQS_FULL_C("DDR3 - DQS Find Limits - NewUnlockPup: ",
|
|
new_unlock_pup, 2);
|
|
|
|
/* Update pup state */
|
|
for (pup = 0; pup < max_pup; pup++) {
|
|
if (IS_PUP_ACTIVE(unlock_pup, pup) == 0) {
|
|
DEBUG_DQS_FULL_C("DDR3 - DQS Find Limits - Skipping pup ",
|
|
pup, 1);
|
|
continue;
|
|
}
|
|
|
|
/*
|
|
* Still didn't find the window limit of the pup
|
|
*/
|
|
if (IS_PUP_ACTIVE(new_unlock_pup, pup) == 1) {
|
|
/* Current compare result == fail */
|
|
if (pup_adll_limit_state[pup] ==
|
|
PUP_ADLL_LIMITS_STATE_PASS) {
|
|
/*
|
|
* If now it failed but passed
|
|
* earlier
|
|
*/
|
|
DEBUG_DQS_S("DDR3 - DQS Find Limits - PASS to FAIL: CS - ");
|
|
DEBUG_DQS_D(cs, 1);
|
|
DEBUG_DQS_S(", DQ - ");
|
|
DEBUG_DQS_D(victim_dq, 1);
|
|
DEBUG_DQS_S(", Pup - ");
|
|
DEBUG_DQS_D(pup, 1);
|
|
DEBUG_DQS_S(", ADLL - ");
|
|
DEBUG_DQS_D(curr_adll, 2);
|
|
DEBUG_DQS_S("\n");
|
|
|
|
#if defined(MV88F78X60)
|
|
for (dq = 0; dq < DQ_NUM; dq++) {
|
|
if ((analog_pbs[victim_dq][pup][dq][0] != adll_start_val)
|
|
&& (analog_pbs[victim_dq][pup]
|
|
[dq][1] == adll_end_val))
|
|
analog_pbs
|
|
[victim_dq]
|
|
[pup][dq]
|
|
[1] =
|
|
curr_adll;
|
|
}
|
|
#endif
|
|
win_valid = 1;
|
|
update_win = 0;
|
|
|
|
/* Keep min / max limit value */
|
|
if (is_tx == 0) {
|
|
/* RX - found upper limit */
|
|
if (centralization_high_limit[pup] >
|
|
(curr_adll - 1)) {
|
|
high_limit =
|
|
curr_adll - 1;
|
|
low_limit =
|
|
centralization_low_limit[pup];
|
|
update_win = 1;
|
|
}
|
|
} else {
|
|
/* TX - found lower limit */
|
|
if (centralization_low_limit[pup] < (curr_adll + 1)) {
|
|
high_limit =
|
|
centralization_high_limit
|
|
[pup];
|
|
low_limit =
|
|
curr_adll + 1;
|
|
update_win =
|
|
1;
|
|
}
|
|
}
|
|
|
|
if (update_win == 1) {
|
|
/*
|
|
* Before updating
|
|
* window limits we need
|
|
* to check that the
|
|
* limits are valid
|
|
*/
|
|
if (MV_OK !=
|
|
ddr3_check_window_limits
|
|
(pup, high_limit,
|
|
low_limit, is_tx,
|
|
&win_valid))
|
|
return MV_DDR3_TRAINING_ERR_WIN_LIMITS;
|
|
|
|
if (win_valid == 1) {
|
|
/*
|
|
* Window limits
|
|
* should be
|
|
* updated
|
|
*/
|
|
centralization_low_limit
|
|
[pup] =
|
|
low_limit;
|
|
centralization_high_limit
|
|
[pup] =
|
|
high_limit;
|
|
}
|
|
}
|
|
|
|
if (win_valid == 1) {
|
|
/* Found end of window - lock the pup */
|
|
pup_adll_limit_state[pup] =
|
|
PUP_ADLL_LIMITS_STATE_FAIL_AFTER_PASS;
|
|
unlock_pup &= ~(1 << pup);
|
|
} else {
|
|
/* Probably false pass - reset status */
|
|
pup_adll_limit_state[pup] =
|
|
PUP_ADLL_LIMITS_STATE_FAIL;
|
|
|
|
#if defined(MV88F78X60)
|
|
/* Clear logging array of win size (per Dq) */
|
|
for (dq = 0;
|
|
dq < DQ_NUM;
|
|
dq++) {
|
|
analog_pbs
|
|
[victim_dq]
|
|
[pup][dq]
|
|
[0] =
|
|
adll_start_val;
|
|
analog_pbs
|
|
[victim_dq]
|
|
[pup][dq]
|
|
[1] =
|
|
adll_end_val;
|
|
per_bit_data
|
|
[pup][dq]
|
|
= 0;
|
|
}
|
|
#endif
|
|
}
|
|
}
|
|
} else {
|
|
/* Current compare result == pass */
|
|
if (pup_adll_limit_state[pup] ==
|
|
PUP_ADLL_LIMITS_STATE_FAIL) {
|
|
/* If now it passed but failed earlier */
|
|
DEBUG_DQS_S("DDR3 - DQS Find Limits - FAIL to PASS: CS - ");
|
|
DEBUG_DQS_D(cs, 1);
|
|
DEBUG_DQS_S(", DQ - ");
|
|
DEBUG_DQS_D(victim_dq, 1);
|
|
DEBUG_DQS_S(", Pup - ");
|
|
DEBUG_DQS_D(pup, 1);
|
|
DEBUG_DQS_S(", ADLL - ");
|
|
DEBUG_DQS_D(curr_adll, 2);
|
|
DEBUG_DQS_S("\n");
|
|
|
|
#if defined(MV88F78X60)
|
|
for (dq = 0; dq < DQ_NUM;
|
|
dq++) {
|
|
if (analog_pbs[victim_dq][pup][dq][0] == adll_start_val)
|
|
analog_pbs
|
|
[victim_dq]
|
|
[pup][dq]
|
|
[0] =
|
|
curr_adll;
|
|
}
|
|
#endif
|
|
/* Found start of window */
|
|
pup_adll_limit_state[pup] =
|
|
PUP_ADLL_LIMITS_STATE_PASS;
|
|
|
|
/* Keep min / max limit value */
|
|
if (is_tx == 0) {
|
|
/* RX - found low limit */
|
|
if (centralization_low_limit[pup] <= curr_adll)
|
|
centralization_low_limit
|
|
[pup] =
|
|
curr_adll;
|
|
} else {
|
|
/* TX - found high limit */
|
|
if (centralization_high_limit[pup] >= curr_adll)
|
|
centralization_high_limit
|
|
[pup] =
|
|
curr_adll;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (unlock_pup == 0) {
|
|
/* Found limit to all pups */
|
|
DEBUG_DQS_FULL_S("DDR3 - DQS Find Limits - found PUP limit\n");
|
|
break;
|
|
}
|
|
|
|
/*
|
|
* Increment / decrement (Move to right / left
|
|
* one phase - ADLL) dqs RX / TX delay (for all un
|
|
* lock pups
|
|
*/
|
|
if (is_tx == 0)
|
|
curr_adll++;
|
|
else
|
|
curr_adll--;
|
|
}
|
|
|
|
if (unlock_pup != 0) {
|
|
/*
|
|
* Found pups that didn't reach to the end of the
|
|
* state machine
|
|
*/
|
|
DEBUG_DQS_C("DDR3 - DQS Find Limits - Pups that didn't reached end of the state machine: ",
|
|
unlock_pup, 1);
|
|
|
|
for (pup = 0; pup < max_pup; pup++) {
|
|
if (IS_PUP_ACTIVE(unlock_pup, pup) == 1) {
|
|
if (pup_adll_limit_state[pup] ==
|
|
PUP_ADLL_LIMITS_STATE_FAIL) {
|
|
/* ERROR - found fail for all window size */
|
|
DEBUG_DQS_S("DDR3 - DQS Find Limits - Got FAIL for the complete range on pup - ");
|
|
DEBUG_DQS_D(pup, 1);
|
|
DEBUG_DQS_C(" victim DQ ",
|
|
victim_dq, 1);
|
|
|
|
/* For debug - set min limit to illegal limit */
|
|
centralization_low_limit[pup]
|
|
= ADLL_ERROR;
|
|
/*
|
|
* In case the pup is in mode
|
|
* PASS - the limit is the min
|
|
* / max adll, no need to
|
|
* update because of the results
|
|
* array default value
|
|
*/
|
|
return MV_DDR3_TRAINING_ERR_PUP_RANGE;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
DEBUG_DQS_S("DDR3 - DQS Find Limits - DQ values per victim results:\n");
|
|
for (victim_dq = 0; victim_dq < DQ_NUM; victim_dq++) {
|
|
for (pup = 0; pup < max_pup; pup++) {
|
|
DEBUG_DQS_S("Victim DQ-");
|
|
DEBUG_DQS_D(victim_dq, 1);
|
|
DEBUG_DQS_S(", PUP-");
|
|
DEBUG_DQS_D(pup, 1);
|
|
for (dq = 0; dq < DQ_NUM; dq++) {
|
|
DEBUG_DQS_S(", DQ-");
|
|
DEBUG_DQS_D(dq, 1);
|
|
DEBUG_DQS_S(",S-");
|
|
DEBUG_DQS_D(analog_pbs[victim_dq][pup][dq]
|
|
[0], 2);
|
|
DEBUG_DQS_S(",E-");
|
|
DEBUG_DQS_D(analog_pbs[victim_dq][pup][dq]
|
|
[1], 2);
|
|
|
|
if (is_tx == 0) {
|
|
if (analog_pbs[victim_dq][pup][dq][0]
|
|
> analog_pbs_sum[pup][dq][0])
|
|
analog_pbs_sum[pup][dq][0] =
|
|
analog_pbs[victim_dq][pup]
|
|
[dq][0];
|
|
if (analog_pbs[victim_dq][pup][dq][1]
|
|
< analog_pbs_sum[pup][dq][1])
|
|
analog_pbs_sum[pup][dq][1] =
|
|
analog_pbs[victim_dq][pup]
|
|
[dq][1];
|
|
} else {
|
|
if (analog_pbs[victim_dq][pup][dq][0]
|
|
< analog_pbs_sum[pup][dq][0])
|
|
analog_pbs_sum[pup][dq][0] =
|
|
analog_pbs[victim_dq][pup]
|
|
[dq][0];
|
|
if (analog_pbs[victim_dq][pup][dq][1]
|
|
> analog_pbs_sum[pup][dq][1])
|
|
analog_pbs_sum[pup][dq][1] =
|
|
analog_pbs[victim_dq][pup]
|
|
[dq][1];
|
|
}
|
|
}
|
|
DEBUG_DQS_S("\n");
|
|
}
|
|
}
|
|
|
|
if (ddr3_get_log_level() >= MV_LOG_LEVEL_3) {
|
|
u32 dq;
|
|
|
|
DEBUG_PER_DQ_S("\n########## LOG LEVEL 3(Windows margins per-DQ) ##########\n");
|
|
if (is_tx) {
|
|
DEBUG_PER_DQ_C("DDR3 - TX CS: ", cs, 1);
|
|
} else {
|
|
DEBUG_PER_DQ_C("DDR3 - RX CS: ", cs, 1);
|
|
}
|
|
|
|
if (ecc == 0) {
|
|
DEBUG_PER_DQ_S("\n DATA RESULTS:\n");
|
|
} else {
|
|
DEBUG_PER_DQ_S("\n ECC RESULTS:\n");
|
|
}
|
|
|
|
/* Since all dq has the same value we take 0 as representive */
|
|
dq = 0;
|
|
for (pup = 0; pup < max_pup; pup++) {
|
|
if (ecc == 0) {
|
|
DEBUG_PER_DQ_S("\nBYTE:");
|
|
DEBUG_PER_DQ_D(pup, 1);
|
|
DEBUG_PER_DQ_S("\n");
|
|
} else {
|
|
DEBUG_PER_DQ_S("\nECC BYTE:\n");
|
|
}
|
|
DEBUG_PER_DQ_S(" DQ's LOW HIGH WIN-SIZE\n");
|
|
DEBUG_PER_DQ_S("============================================\n");
|
|
for (victim_dq = 0; victim_dq < DQ_NUM; victim_dq++) {
|
|
if (ecc == 0) {
|
|
DEBUG_PER_DQ_S("DQ[");
|
|
DEBUG_PER_DQ_DD((victim_dq +
|
|
DQ_NUM * pup), 2);
|
|
DEBUG_PER_DQ_S("]");
|
|
} else {
|
|
DEBUG_PER_DQ_S("CB[");
|
|
DEBUG_PER_DQ_DD(victim_dq, 2);
|
|
DEBUG_PER_DQ_S("]");
|
|
}
|
|
if (is_tx) {
|
|
DEBUG_PER_DQ_S(" 0x");
|
|
DEBUG_PER_DQ_D(analog_pbs[victim_dq][pup][dq][1], 2); /* low value */
|
|
DEBUG_PER_DQ_S(" 0x");
|
|
DEBUG_PER_DQ_D(analog_pbs[victim_dq][pup][dq][0], 2); /* high value */
|
|
DEBUG_PER_DQ_S(" 0x");
|
|
DEBUG_PER_DQ_D(analog_pbs[victim_dq][pup][dq][0] - analog_pbs[victim_dq][pup][dq][1], 2); /* win-size */
|
|
} else {
|
|
DEBUG_PER_DQ_S(" 0x");
|
|
DEBUG_PER_DQ_D(analog_pbs[victim_dq][pup][dq][0], 2); /* low value */
|
|
DEBUG_PER_DQ_S(" 0x");
|
|
DEBUG_PER_DQ_D((analog_pbs[victim_dq][pup][dq][1] - 1), 2); /* high value */
|
|
DEBUG_PER_DQ_S(" 0x");
|
|
DEBUG_PER_DQ_D(analog_pbs[victim_dq][pup][dq][1] - analog_pbs[victim_dq][pup][dq][0], 2); /* win-size */
|
|
}
|
|
DEBUG_PER_DQ_S("\n");
|
|
}
|
|
}
|
|
DEBUG_PER_DQ_S("\n");
|
|
}
|
|
|
|
if (is_tx) {
|
|
DEBUG_DQS_S("DDR3 - DQS TX - Find Limits - DQ values Summary:\n");
|
|
} else {
|
|
DEBUG_DQS_S("DDR3 - DQS RX - Find Limits - DQ values Summary:\n");
|
|
}
|
|
|
|
for (pup = 0; pup < max_pup; pup++) {
|
|
DEBUG_DQS_S("PUP-");
|
|
DEBUG_DQS_D(pup, 1);
|
|
for (dq = 0; dq < DQ_NUM; dq++) {
|
|
DEBUG_DQS_S(", DQ-");
|
|
DEBUG_DQS_D(dq, 1);
|
|
DEBUG_DQS_S(",S-");
|
|
DEBUG_DQS_D(analog_pbs_sum[pup][dq][0], 2);
|
|
DEBUG_DQS_S(",E-");
|
|
DEBUG_DQS_D(analog_pbs_sum[pup][dq][1], 2);
|
|
}
|
|
DEBUG_DQS_S("\n");
|
|
}
|
|
|
|
if (is_tx) {
|
|
DEBUG_DQS_S("DDR3 - DQS TX - Find Limits - DQ values Summary:\n");
|
|
} else {
|
|
DEBUG_DQS_S("DDR3 - DQS RX - Find Limits - DQ values Summary:\n");
|
|
}
|
|
|
|
for (pup = 0; pup < max_pup; pup++) {
|
|
if (max_pup == 1) {
|
|
/* For ECC PUP */
|
|
DEBUG_DQS_S("DDR3 - DQS8");
|
|
} else {
|
|
DEBUG_DQS_S("DDR3 - DQS");
|
|
DEBUG_DQS_D(pup, 1);
|
|
}
|
|
|
|
for (dq = 0; dq < DQ_NUM; dq++) {
|
|
DEBUG_DQS_S(", DQ-");
|
|
DEBUG_DQS_D(dq, 1);
|
|
DEBUG_DQS_S("::S-");
|
|
DEBUG_DQS_D(analog_pbs_sum[pup][dq][0], 2);
|
|
DEBUG_DQS_S(",E-");
|
|
DEBUG_DQS_D(analog_pbs_sum[pup][dq][1], 2);
|
|
}
|
|
DEBUG_DQS_S("\n");
|
|
}
|
|
|
|
DEBUG_DQS_S("DDR3 - DQS Find Limits - Ended\n");
|
|
|
|
return MV_OK;
|
|
}
|
|
|
|
/*
|
|
* Name: ddr3_check_window_limits
|
|
* Desc: Check window High & Low limits.
|
|
* Args: pup pup index
|
|
* high_limit window high limit
|
|
* low_limit window low limit
|
|
* is_tx Indicate whether Rx or Tx
|
|
* size_valid Indicate whether window size is valid
|
|
* Notes:
|
|
* Returns: MV_OK if success, other error code if fail.
|
|
*/
|
|
int ddr3_check_window_limits(u32 pup, int high_limit, int low_limit, int is_tx,
|
|
int *size_valid)
|
|
{
|
|
DEBUG_DQS_FULL_S("DDR3 - DQS Check Win Limits - Starting\n");
|
|
|
|
if (low_limit > high_limit) {
|
|
DEBUG_DQS_S("DDR3 - DQS Check Win Limits - Pup ");
|
|
DEBUG_DQS_D(pup, 1);
|
|
DEBUG_DQS_S(" Low Limit grater than High Limit\n");
|
|
*size_valid = 0;
|
|
return MV_OK;
|
|
}
|
|
|
|
/*
|
|
* Check that window size is valid, if not it was probably false pass
|
|
* before
|
|
*/
|
|
if ((high_limit - low_limit) < MIN_WIN_SIZE) {
|
|
/*
|
|
* Since window size is too small probably there was false
|
|
* pass
|
|
*/
|
|
*size_valid = 0;
|
|
|
|
DEBUG_DQS_S("DDR3 - DQS Check Win Limits - Pup ");
|
|
DEBUG_DQS_D(pup, 1);
|
|
DEBUG_DQS_S(" Window size is smaller than MIN_WIN_SIZE\n");
|
|
|
|
} else if ((high_limit - low_limit) > ADLL_MAX) {
|
|
*size_valid = 0;
|
|
|
|
DEBUG_DQS_S("DDR3 - DQS Check Win Limits - Pup ");
|
|
DEBUG_DQS_D(pup, 1);
|
|
DEBUG_DQS_S
|
|
(" Window size is bigger than max ADLL taps (31) Exiting.\n");
|
|
|
|
return MV_FAIL;
|
|
|
|
} else {
|
|
*size_valid = 1;
|
|
|
|
DEBUG_DQS_FULL_S("DDR3 - DQS Check Win Limits - Pup ");
|
|
DEBUG_DQS_FULL_D(pup, 1);
|
|
DEBUG_DQS_FULL_C(" window size is ", (high_limit - low_limit),
|
|
2);
|
|
}
|
|
|
|
return MV_OK;
|
|
}
|
|
|
|
/*
|
|
* Name: ddr3_center_calc
|
|
* Desc: Execute the calculate the center of windows phase.
|
|
* Args: pDram Info
|
|
* is_tx Indicate whether Rx or Tx
|
|
* Notes:
|
|
* Returns: MV_OK if success, other error code if fail.
|
|
*/
|
|
static int ddr3_center_calc(MV_DRAM_INFO *dram_info, u32 cs, u32 ecc,
|
|
int is_tx)
|
|
{
|
|
/* bit array of pups that need specail search */
|
|
u32 special_pattern_i_pup = 0;
|
|
u32 special_pattern_ii_pup = 0;
|
|
u32 pup;
|
|
u32 max_pup;
|
|
|
|
max_pup = (ecc + (1 - ecc) * dram_info->num_of_std_pups);
|
|
|
|
for (pup = 0; pup < max_pup; pup++) {
|
|
if (is_tx == 0) {
|
|
/* Check special pattern I */
|
|
/*
|
|
* Special pattern Low limit search - relevant only
|
|
* for Rx, win size < threshold and low limit = 0
|
|
*/
|
|
if (((centralization_high_limit[pup] -
|
|
centralization_low_limit[pup]) < VALID_WIN_THRS)
|
|
&& (centralization_low_limit[pup] == MIN_DELAY))
|
|
special_pattern_i_pup |= (1 << pup);
|
|
|
|
/* Check special pattern II */
|
|
/*
|
|
* Special pattern High limit search - relevant only
|
|
* for Rx, win size < threshold and high limit = 31
|
|
*/
|
|
if (((centralization_high_limit[pup] -
|
|
centralization_low_limit[pup]) < VALID_WIN_THRS)
|
|
&& (centralization_high_limit[pup] == MAX_DELAY))
|
|
special_pattern_ii_pup |= (1 << pup);
|
|
}
|
|
}
|
|
|
|
/* Run special pattern Low limit search - for relevant pup */
|
|
if (special_pattern_i_pup != 0) {
|
|
DEBUG_DQS_S("DDR3 - DQS Center Calc - Entering special pattern I for Low limit search\n");
|
|
if (MV_OK !=
|
|
ddr3_special_pattern_i_search(dram_info, cs, ecc, is_tx,
|
|
special_pattern_i_pup))
|
|
return MV_DDR3_TRAINING_ERR_DQS_LOW_LIMIT_SEARCH;
|
|
}
|
|
|
|
/* Run special pattern High limit search - for relevant pup */
|
|
if (special_pattern_ii_pup != 0) {
|
|
DEBUG_DQS_S("DDR3 - DQS Center Calc - Entering special pattern II for High limit search\n");
|
|
if (MV_OK !=
|
|
ddr3_special_pattern_ii_search(dram_info, cs, ecc, is_tx,
|
|
special_pattern_ii_pup))
|
|
return MV_DDR3_TRAINING_ERR_DQS_HIGH_LIMIT_SEARCH;
|
|
}
|
|
|
|
/* Set adll to center = (General_High_limit + General_Low_limit)/2 */
|
|
return ddr3_set_dqs_centralization_results(dram_info, cs, ecc, is_tx);
|
|
}
|
|
|
|
/*
|
|
* Name: ddr3_special_pattern_i_search
|
|
* Desc: Execute special pattern low limit search.
|
|
* Args:
|
|
* special_pattern_pup The pups that need the special search
|
|
* Notes:
|
|
* Returns: MV_OK if success, other error code if fail.
|
|
*/
|
|
int ddr3_special_pattern_i_search(MV_DRAM_INFO *dram_info, u32 cs, u32 ecc,
|
|
int is_tx, u32 special_pattern_pup)
|
|
{
|
|
u32 victim_dq; /* loop index - victim DQ */
|
|
u32 adll_idx;
|
|
u32 pup;
|
|
u32 unlock_pup; /* bit array of the unlock pups */
|
|
u32 first_fail; /* bit array - of pups that get first fail */
|
|
u32 new_lockup_pup; /* bit array of compare failed pups */
|
|
u32 pass_pup; /* bit array of compare pass pup */
|
|
u32 sdram_offset;
|
|
u32 max_pup;
|
|
u32 comp_val;
|
|
u32 special_res[MAX_PUP_NUM]; /* hold tmp results */
|
|
|
|
DEBUG_DQS_S("DDR3 - DQS - Special Pattern I Search - Starting\n");
|
|
|
|
max_pup = ecc + (1 - ecc) * dram_info->num_of_std_pups;
|
|
|
|
/* Init the temporary results to max ADLL value */
|
|
for (pup = 0; pup < max_pup; pup++)
|
|
special_res[pup] = ADLL_MAX;
|
|
|
|
/* Run special pattern for all DQ - use the same pattern */
|
|
for (victim_dq = 0; victim_dq < DQ_NUM; victim_dq++) {
|
|
unlock_pup = special_pattern_pup;
|
|
first_fail = 0;
|
|
|
|
sdram_offset = cs * SDRAM_CS_SIZE + SDRAM_DQS_RX_OFFS +
|
|
LEN_KILLER_PATTERN * 4 * victim_dq;
|
|
|
|
for (pup = 0; pup < max_pup; pup++) {
|
|
/* Set adll value per PUP. adll = high limit per pup */
|
|
if (IS_PUP_ACTIVE(unlock_pup, pup)) {
|
|
/* only for pups that need special search */
|
|
ddr3_write_pup_reg(PUP_DQS_RD, cs,
|
|
pup + (ecc * ECC_PUP), 0,
|
|
centralization_high_limit
|
|
[pup]);
|
|
}
|
|
}
|
|
|
|
adll_idx = 0;
|
|
do {
|
|
/*
|
|
* Perform read and compare simultaneously for all
|
|
* un-locked MC use the special pattern mask
|
|
*/
|
|
new_lockup_pup = 0;
|
|
|
|
if (MV_OK !=
|
|
ddr3_sdram_dqs_compare(dram_info, unlock_pup,
|
|
&new_lockup_pup,
|
|
special_pattern
|
|
[victim_dq],
|
|
LEN_SPECIAL_PATTERN,
|
|
sdram_offset, 0,
|
|
0, NULL, 1))
|
|
return MV_FAIL;
|
|
|
|
DEBUG_DQS_S("DDR3 - DQS - Special I - ADLL value is: ");
|
|
DEBUG_DQS_D(adll_idx, 2);
|
|
DEBUG_DQS_S(", UnlockPup: ");
|
|
DEBUG_DQS_D(unlock_pup, 2);
|
|
DEBUG_DQS_S(", NewLockPup: ");
|
|
DEBUG_DQS_D(new_lockup_pup, 2);
|
|
DEBUG_DQS_S("\n");
|
|
|
|
if (unlock_pup != new_lockup_pup)
|
|
DEBUG_DQS_S("DDR3 - DQS - Special I - Some Pup passed!\n");
|
|
|
|
/* Search for pups with passed compare & already fail */
|
|
pass_pup = first_fail & ~new_lockup_pup & unlock_pup;
|
|
first_fail |= new_lockup_pup;
|
|
unlock_pup &= ~pass_pup;
|
|
|
|
/* Get pass pups */
|
|
if (pass_pup != 0) {
|
|
for (pup = 0; pup < max_pup; pup++) {
|
|
if (IS_PUP_ACTIVE(pass_pup, pup) ==
|
|
1) {
|
|
/* If pup passed and has first fail = 1 */
|
|
/* keep min value of ADLL max value - current adll */
|
|
/* (centralization_high_limit[pup] + adll_idx) = current adll !!! */
|
|
comp_val =
|
|
(ADLL_MAX -
|
|
(centralization_high_limit
|
|
[pup] + adll_idx));
|
|
|
|
DEBUG_DQS_C
|
|
("DDR3 - DQS - Special I - Pup - ",
|
|
pup, 1);
|
|
DEBUG_DQS_C
|
|
(" comp_val = ",
|
|
comp_val, 2);
|
|
|
|
if (comp_val <
|
|
special_res[pup]) {
|
|
special_res[pup] =
|
|
comp_val;
|
|
centralization_low_limit
|
|
[pup] =
|
|
(-1) *
|
|
comp_val;
|
|
|
|
DEBUG_DQS_C
|
|
("DDR3 - DQS - Special I - Pup - ",
|
|
pup, 1);
|
|
DEBUG_DQS_C
|
|
(" Changed Low limit to ",
|
|
centralization_low_limit
|
|
[pup], 2);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Did all PUP found missing window?
|
|
* Check for each pup if adll (different for each pup)
|
|
* reach maximum if reach max value - lock the pup
|
|
* if not - increment (Move to right one phase - ADLL)
|
|
* dqs RX delay
|
|
*/
|
|
adll_idx++;
|
|
for (pup = 0; pup < max_pup; pup++) {
|
|
if (IS_PUP_ACTIVE(unlock_pup, pup) == 1) {
|
|
/* Check only unlocked pups */
|
|
if ((centralization_high_limit[pup] +
|
|
adll_idx) >= ADLL_MAX) {
|
|
/* reach maximum - lock the pup */
|
|
DEBUG_DQS_C("DDR3 - DQS - Special I - reach maximum - lock pup ",
|
|
pup, 1);
|
|
unlock_pup &= ~(1 << pup);
|
|
} else {
|
|
/* Didn't reach maximum - increment ADLL */
|
|
ddr3_write_pup_reg(PUP_DQS_RD,
|
|
cs,
|
|
pup +
|
|
(ecc *
|
|
ECC_PUP), 0,
|
|
(centralization_high_limit
|
|
[pup] +
|
|
adll_idx));
|
|
}
|
|
}
|
|
}
|
|
} while (unlock_pup != 0);
|
|
}
|
|
|
|
return MV_OK;
|
|
}
|
|
|
|
/*
|
|
* Name: ddr3_special_pattern_ii_search
|
|
* Desc: Execute special pattern high limit search.
|
|
* Args:
|
|
* special_pattern_pup The pups that need the special search
|
|
* Notes:
|
|
* Returns: MV_OK if success, other error code if fail.
|
|
*/
|
|
int ddr3_special_pattern_ii_search(MV_DRAM_INFO *dram_info, u32 cs, u32 ecc,
|
|
int is_tx, u32 special_pattern_pup)
|
|
{
|
|
u32 victim_dq; /* loop index - victim DQ */
|
|
u32 adll_idx;
|
|
u32 pup;
|
|
u32 unlock_pup; /* bit array of the unlock pups */
|
|
u32 first_fail; /* bit array - of pups that get first fail */
|
|
u32 new_lockup_pup; /* bit array of compare failed pups */
|
|
u32 pass_pup; /* bit array of compare pass pup */
|
|
u32 sdram_offset;
|
|
u32 max_pup;
|
|
u32 comp_val;
|
|
u32 special_res[MAX_PUP_NUM]; /* hold tmp results */
|
|
|
|
DEBUG_DQS_S("DDR3 - DQS - Special Pattern II Search - Starting\n");
|
|
|
|
max_pup = (ecc + (1 - ecc) * dram_info->num_of_std_pups);
|
|
|
|
/* init the tmporary results to max ADLL value */
|
|
for (pup = 0; pup < max_pup; pup++)
|
|
special_res[pup] = ADLL_MAX;
|
|
|
|
sdram_offset = cs * SDRAM_CS_SIZE + SDRAM_DQS_RX_OFFS;
|
|
|
|
/* run special pattern for all DQ - use the same pattern */
|
|
for (victim_dq = 0; victim_dq < DQ_NUM; victim_dq++) {
|
|
unlock_pup = special_pattern_pup;
|
|
first_fail = 0;
|
|
|
|
for (pup = 0; pup < max_pup; pup++) {
|
|
/* Set adll value per PUP. adll = 0 */
|
|
if (IS_PUP_ACTIVE(unlock_pup, pup)) {
|
|
/* Only for pups that need special search */
|
|
ddr3_write_pup_reg(PUP_DQS_RD, cs,
|
|
pup + (ecc * ECC_PUP), 0,
|
|
ADLL_MIN);
|
|
}
|
|
}
|
|
|
|
adll_idx = 0;
|
|
do {
|
|
/*
|
|
* Perform read and compare simultaneously for all
|
|
* un-locked MC use the special pattern mask
|
|
*/
|
|
new_lockup_pup = 0;
|
|
|
|
if (MV_OK != ddr3_sdram_dqs_compare(
|
|
dram_info, unlock_pup, &new_lockup_pup,
|
|
special_pattern[victim_dq],
|
|
LEN_SPECIAL_PATTERN,
|
|
sdram_offset, 0, 0, NULL, 0))
|
|
return MV_FAIL;
|
|
|
|
DEBUG_DQS_S("DDR3 - DQS - Special II - ADLL value is ");
|
|
DEBUG_DQS_D(adll_idx, 2);
|
|
DEBUG_DQS_S("unlock_pup ");
|
|
DEBUG_DQS_D(unlock_pup, 1);
|
|
DEBUG_DQS_S("new_lockup_pup ");
|
|
DEBUG_DQS_D(new_lockup_pup, 1);
|
|
DEBUG_DQS_S("\n");
|
|
|
|
if (unlock_pup != new_lockup_pup) {
|
|
DEBUG_DQS_S("DDR3 - DQS - Special II - Some Pup passed!\n");
|
|
}
|
|
|
|
/* Search for pups with passed compare & already fail */
|
|
pass_pup = first_fail & ~new_lockup_pup & unlock_pup;
|
|
first_fail |= new_lockup_pup;
|
|
unlock_pup &= ~pass_pup;
|
|
|
|
/* Get pass pups */
|
|
if (pass_pup != 0) {
|
|
for (pup = 0; pup < max_pup; pup++) {
|
|
if (IS_PUP_ACTIVE(pass_pup, pup) ==
|
|
1) {
|
|
/* If pup passed and has first fail = 1 */
|
|
/* keep min value of ADLL max value - current adll */
|
|
/* (adll_idx) = current adll !!! */
|
|
comp_val = adll_idx;
|
|
|
|
DEBUG_DQS_C("DDR3 - DQS - Special II - Pup - ",
|
|
pup, 1);
|
|
DEBUG_DQS_C(" comp_val = ",
|
|
comp_val, 1);
|
|
|
|
if (comp_val <
|
|
special_res[pup]) {
|
|
special_res[pup] =
|
|
comp_val;
|
|
centralization_high_limit
|
|
[pup] =
|
|
ADLL_MAX +
|
|
comp_val;
|
|
|
|
DEBUG_DQS_C
|
|
("DDR3 - DQS - Special II - Pup - ",
|
|
pup, 1);
|
|
DEBUG_DQS_C
|
|
(" Changed High limit to ",
|
|
centralization_high_limit
|
|
[pup], 2);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Did all PUP found missing window?
|
|
* Check for each pup if adll (different for each pup)
|
|
* reach maximum if reach max value - lock the pup
|
|
* if not - increment (Move to right one phase - ADLL)
|
|
* dqs RX delay
|
|
*/
|
|
adll_idx++;
|
|
for (pup = 0; pup < max_pup; pup++) {
|
|
if (IS_PUP_ACTIVE(unlock_pup, pup) == 1) {
|
|
/* Check only unlocked pups */
|
|
if ((adll_idx) >= ADLL_MAX) {
|
|
/* Reach maximum - lock the pup */
|
|
DEBUG_DQS_C("DDR3 - DQS - Special II - reach maximum - lock pup ",
|
|
pup, 1);
|
|
unlock_pup &= ~(1 << pup);
|
|
} else {
|
|
/* Didn't reach maximum - increment ADLL */
|
|
ddr3_write_pup_reg(PUP_DQS_RD,
|
|
cs,
|
|
pup +
|
|
(ecc *
|
|
ECC_PUP), 0,
|
|
(adll_idx));
|
|
}
|
|
}
|
|
}
|
|
} while (unlock_pup != 0);
|
|
}
|
|
|
|
return MV_OK;
|
|
}
|
|
|
|
/*
|
|
* Name: ddr3_set_dqs_centralization_results
|
|
* Desc: Set to HW the DQS centralization phase results.
|
|
* Args:
|
|
* is_tx Indicates whether to set Tx or RX results
|
|
* Notes:
|
|
* Returns: MV_OK if success, other error code if fail.
|
|
*/
|
|
int ddr3_set_dqs_centralization_results(MV_DRAM_INFO *dram_info, u32 cs,
|
|
u32 ecc, int is_tx)
|
|
{
|
|
u32 pup, pup_num;
|
|
int addl_val;
|
|
u32 max_pup;
|
|
|
|
max_pup = (ecc + (1 - ecc) * dram_info->num_of_std_pups);
|
|
|
|
DEBUG_DQS_RESULTS_S("\n############ LOG LEVEL 2(Windows margins) ############\n");
|
|
|
|
if (is_tx) {
|
|
DEBUG_DQS_RESULTS_C("DDR3 - DQS TX - Set Dqs Centralization Results - CS: ",
|
|
cs, 1);
|
|
} else {
|
|
DEBUG_DQS_RESULTS_C("DDR3 - DQS RX - Set Dqs Centralization Results - CS: ",
|
|
cs, 1);
|
|
}
|
|
|
|
/* Set adll to center = (General_High_limit + General_Low_limit)/2 */
|
|
DEBUG_DQS_RESULTS_S("\nDQS LOW HIGH WIN-SIZE Set\n");
|
|
DEBUG_DQS_RESULTS_S("==============================================\n");
|
|
for (pup = 0; pup < max_pup; pup++) {
|
|
addl_val = (centralization_high_limit[pup] +
|
|
centralization_low_limit[pup]) / 2;
|
|
|
|
pup_num = pup * (1 - ecc) + ecc * ECC_PUP;
|
|
|
|
DEBUG_DQS_RESULTS_D(pup_num, 1);
|
|
DEBUG_DQS_RESULTS_S(" 0x");
|
|
DEBUG_DQS_RESULTS_D(centralization_low_limit[pup], 2);
|
|
DEBUG_DQS_RESULTS_S(" 0x");
|
|
DEBUG_DQS_RESULTS_D(centralization_high_limit[pup], 2);
|
|
DEBUG_DQS_RESULTS_S(" 0x");
|
|
DEBUG_DQS_RESULTS_D(centralization_high_limit[pup] -
|
|
centralization_low_limit[pup], 2);
|
|
DEBUG_DQS_RESULTS_S(" 0x");
|
|
DEBUG_DQS_RESULTS_D(addl_val, 2);
|
|
DEBUG_DQS_RESULTS_S("\n");
|
|
|
|
if (addl_val < ADLL_MIN) {
|
|
addl_val = ADLL_MIN;
|
|
DEBUG_DQS_RESULTS_S("DDR3 - DQS - Setting ADLL value for Pup to MIN (since it was lower than 0)\n");
|
|
}
|
|
|
|
if (addl_val > ADLL_MAX) {
|
|
addl_val = ADLL_MAX;
|
|
DEBUG_DQS_RESULTS_S("DDR3 - DQS - Setting ADLL value for Pup to MAX (since it was higher than 31)\n");
|
|
}
|
|
|
|
if (is_tx) {
|
|
ddr3_write_pup_reg(PUP_DQS_WR, cs, pup_num, 0,
|
|
addl_val +
|
|
dram_info->wl_val[cs][pup_num][D]);
|
|
} else {
|
|
ddr3_write_pup_reg(PUP_DQS_RD, cs, pup_num, 0,
|
|
addl_val);
|
|
}
|
|
}
|
|
|
|
return MV_OK;
|
|
}
|
|
|
|
/*
|
|
* Set training patterns
|
|
*/
|
|
int ddr3_load_dqs_patterns(MV_DRAM_INFO *dram_info)
|
|
{
|
|
u32 cs, cs_count, cs_tmp, victim_dq;
|
|
u32 sdram_addr;
|
|
u32 *pattern_ptr;
|
|
|
|
/* 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 killer pattern */
|
|
sdram_addr = (cs_count * (SDRAM_CS_SIZE + 1) +
|
|
SDRAM_DQS_RX_OFFS);
|
|
for (victim_dq = 0; victim_dq < DQ_NUM; victim_dq++) {
|
|
pattern_ptr = ddr3_dqs_choose_pattern(dram_info,
|
|
victim_dq);
|
|
if (MV_OK != ddr3_sdram_dqs_compare(
|
|
dram_info, (u32)NULL, NULL,
|
|
pattern_ptr, LEN_KILLER_PATTERN,
|
|
sdram_addr + LEN_KILLER_PATTERN *
|
|
4 * victim_dq, 1, 0, NULL,
|
|
0))
|
|
return MV_DDR3_TRAINING_ERR_DQS_PATTERN;
|
|
}
|
|
|
|
/* Init special-killer pattern */
|
|
sdram_addr = (cs_count * (SDRAM_CS_SIZE + 1) +
|
|
SDRAM_DQS_RX_SPECIAL_OFFS);
|
|
for (victim_dq = 0; victim_dq < DQ_NUM; victim_dq++) {
|
|
if (MV_OK != ddr3_sdram_dqs_compare(
|
|
dram_info, (u32)NULL, NULL,
|
|
special_pattern[victim_dq],
|
|
LEN_KILLER_PATTERN, sdram_addr +
|
|
LEN_KILLER_PATTERN * 4 * victim_dq,
|
|
1, 0, NULL, 0))
|
|
return MV_DDR3_TRAINING_ERR_DQS_PATTERN;
|
|
}
|
|
}
|
|
}
|
|
|
|
return MV_OK;
|
|
}
|
|
|