mirror of
https://github.com/enjoy-digital/litex.git
synced 2025-01-04 09:52:26 -05:00
bios/sdram: add write/read leveling scans
This commit is contained in:
parent
34b2bd0c28
commit
b5ee110e63
1 changed files with 94 additions and 1 deletions
|
@ -222,11 +222,41 @@ void sdrwloff(void)
|
||||||
ddrphy_wlevel_en_write(0);
|
ddrphy_wlevel_en_write(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int write_level(int *delay, int *high_skew)
|
static void write_level_scan(void)
|
||||||
{
|
{
|
||||||
int i, j;
|
int i, j;
|
||||||
int dq_address;
|
int dq_address;
|
||||||
unsigned char dq;
|
unsigned char dq;
|
||||||
|
|
||||||
|
printf("Write leveling scan:\n");
|
||||||
|
|
||||||
|
sdrwlon();
|
||||||
|
cdelay(100);
|
||||||
|
for(i=0;i<DFII_PIX_DATA_SIZE/2;i++) {
|
||||||
|
printf("Module %d:\n", i);
|
||||||
|
dq_address = sdram_dfii_pix_rddata_addr[0]+4*(DFII_PIX_DATA_SIZE/2-1-i);
|
||||||
|
ddrphy_dly_sel_write(1 << i);
|
||||||
|
ddrphy_wdly_dq_rst_write(1);
|
||||||
|
ddrphy_wdly_dqs_rst_write(1);
|
||||||
|
for(j=0;j<ERR_DDRPHY_DELAY;j++) {
|
||||||
|
ddrphy_wlevel_strobe_write(1);
|
||||||
|
cdelay(10);
|
||||||
|
dq = MMPTR(dq_address);
|
||||||
|
printf("%d", dq == 0);
|
||||||
|
ddrphy_wdly_dq_inc_write(1);
|
||||||
|
ddrphy_wdly_dqs_inc_write(1);
|
||||||
|
cdelay(10);
|
||||||
|
}
|
||||||
|
printf("\n");
|
||||||
|
}
|
||||||
|
sdrwloff();
|
||||||
|
}
|
||||||
|
|
||||||
|
static int write_level(int *delay, int *high_skew)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
int dq_address;
|
||||||
|
unsigned char dq;
|
||||||
int err_ddrphy_wdly;
|
int err_ddrphy_wdly;
|
||||||
int ok;
|
int ok;
|
||||||
|
|
||||||
|
@ -245,6 +275,7 @@ static int write_level(int *delay, int *high_skew)
|
||||||
ddrphy_wdly_dq_rst_write(1);
|
ddrphy_wdly_dq_rst_write(1);
|
||||||
ddrphy_wdly_dqs_rst_write(1);
|
ddrphy_wdly_dqs_rst_write(1);
|
||||||
#ifdef CSR_DDRPHY_WDLY_DQS_TAPS_ADDR
|
#ifdef CSR_DDRPHY_WDLY_DQS_TAPS_ADDR
|
||||||
|
int j;
|
||||||
for(j=0; j<ddrphy_wdly_dqs_taps_read(); j++)
|
for(j=0; j<ddrphy_wdly_dqs_taps_read(); j++)
|
||||||
ddrphy_wdly_dqs_inc_write(1);
|
ddrphy_wdly_dqs_inc_write(1);
|
||||||
#endif
|
#endif
|
||||||
|
@ -334,6 +365,66 @@ static void read_bitslip(int *delay, int *high_skew)
|
||||||
printf("\n");
|
printf("\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void read_delays_scan(void)
|
||||||
|
{
|
||||||
|
unsigned int prv;
|
||||||
|
unsigned char prs[DFII_NPHASES*DFII_PIX_DATA_SIZE];
|
||||||
|
int p, i, j;
|
||||||
|
int working;
|
||||||
|
|
||||||
|
printf("Read delays scan:\n");
|
||||||
|
|
||||||
|
/* Generate pseudo-random sequence */
|
||||||
|
prv = 42;
|
||||||
|
for(i=0;i<DFII_NPHASES*DFII_PIX_DATA_SIZE;i++) {
|
||||||
|
prv = 1664525*prv + 1013904223;
|
||||||
|
prs[i] = prv;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Activate */
|
||||||
|
sdram_dfii_pi0_address_write(0);
|
||||||
|
sdram_dfii_pi0_baddress_write(0);
|
||||||
|
command_p0(DFII_COMMAND_RAS|DFII_COMMAND_CS);
|
||||||
|
cdelay(15);
|
||||||
|
|
||||||
|
/* Write test pattern */
|
||||||
|
for(p=0;p<DFII_NPHASES;p++)
|
||||||
|
for(i=0;i<DFII_PIX_DATA_SIZE;i++)
|
||||||
|
MMPTR(sdram_dfii_pix_wrdata_addr[p]+4*i) = prs[DFII_PIX_DATA_SIZE*p+i];
|
||||||
|
sdram_dfii_piwr_address_write(0);
|
||||||
|
sdram_dfii_piwr_baddress_write(0);
|
||||||
|
command_pwr(DFII_COMMAND_CAS|DFII_COMMAND_WE|DFII_COMMAND_CS|DFII_COMMAND_WRDATA);
|
||||||
|
|
||||||
|
/* Calibrate each DQ in turn */
|
||||||
|
sdram_dfii_pird_address_write(0);
|
||||||
|
sdram_dfii_pird_baddress_write(0);
|
||||||
|
for(i=0;i<DFII_PIX_DATA_SIZE/2;i++) {
|
||||||
|
printf("Module %d:\n", (DFII_PIX_DATA_SIZE/2-i-1));
|
||||||
|
ddrphy_dly_sel_write(1 << (DFII_PIX_DATA_SIZE/2-i-1));
|
||||||
|
ddrphy_rdly_dq_rst_write(1);
|
||||||
|
for(j=0; j<ERR_DDRPHY_DELAY;j++) {
|
||||||
|
command_prd(DFII_COMMAND_CAS|DFII_COMMAND_CS|DFII_COMMAND_RDDATA);
|
||||||
|
cdelay(15);
|
||||||
|
working = 1;
|
||||||
|
for(p=0;p<DFII_NPHASES;p++) {
|
||||||
|
if(MMPTR(sdram_dfii_pix_rddata_addr[p]+4*i) != prs[DFII_PIX_DATA_SIZE*p+i])
|
||||||
|
working = 0;
|
||||||
|
if(MMPTR(sdram_dfii_pix_rddata_addr[p]+4*(i+DFII_PIX_DATA_SIZE/2)) != prs[DFII_PIX_DATA_SIZE*p+i+DFII_PIX_DATA_SIZE/2])
|
||||||
|
working = 0;
|
||||||
|
}
|
||||||
|
printf("%d", working);
|
||||||
|
ddrphy_rdly_dq_inc_write(1);
|
||||||
|
}
|
||||||
|
printf("\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Precharge */
|
||||||
|
sdram_dfii_pi0_address_write(0);
|
||||||
|
sdram_dfii_pi0_baddress_write(0);
|
||||||
|
command_p0(DFII_COMMAND_RAS|DFII_COMMAND_WE|DFII_COMMAND_CS);
|
||||||
|
cdelay(15);
|
||||||
|
}
|
||||||
|
|
||||||
static void read_delays(void)
|
static void read_delays(void)
|
||||||
{
|
{
|
||||||
unsigned int prv;
|
unsigned int prv;
|
||||||
|
@ -648,10 +739,12 @@ int sdrlevel(void) /* automatic */
|
||||||
high_skew[i] = 0;
|
high_skew[i] = 0;
|
||||||
}
|
}
|
||||||
#else
|
#else
|
||||||
|
write_level_scan();
|
||||||
if(!write_level(delay, high_skew))
|
if(!write_level(delay, high_skew))
|
||||||
return 0;
|
return 0;
|
||||||
#endif
|
#endif
|
||||||
read_bitslip(delay, high_skew);
|
read_bitslip(delay, high_skew);
|
||||||
|
read_delays_scan();
|
||||||
read_delays();
|
read_delays();
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
|
|
Loading…
Reference in a new issue