fsl/ddr: Add workaround for ERRATUM_A009942
authorShengzhou Liu <Shengzhou.Liu@freescale.com>
Wed, 6 Jan 2016 03:26:51 +0000 (11:26 +0800)
committerYork Sun <york.sun@nxp.com>
Mon, 25 Jan 2016 16:24:14 +0000 (08:24 -0800)
During the receive data training, the DDRC may complete on a
non-optimal setting that could lead to data corruption or
initialization failure.

Workaround: before setting MEM_EN, set DEBUG_29 register with
specific value for different data rates.

Signed-off-by: Shengzhou Liu <Shengzhou.Liu@freescale.com>
Reviewed-by: York Sun <york.sun@nxp.com>
arch/arm/include/asm/arch-fsl-layerscape/config.h
drivers/ddr/fsl/fsl_ddr_gen4.c

index 49b113dc59ba8f65ef7a92fbd7044fe22625e83b..83a207c308240ab2d325c548e0caa408b6923855 100644 (file)
 #define CONFIG_SYS_FSL_ERRATUM_A008585
 #define CONFIG_SYS_FSL_ERRATUM_A008751
 #define CONFIG_SYS_FSL_ERRATUM_A009635
+#define CONFIG_SYS_FSL_ERRATUM_A009942
 #elif defined(CONFIG_LS1043A)
 #define CONFIG_MAX_CPUS                                4
 #define CONFIG_SYS_CACHELINE_SIZE              64
index 3fca5c2684bda5025a066bbe5d1751aab43d9b33..0d9dd0bb469c00b98b98590c40093db6b73e837d 100644 (file)
@@ -55,6 +55,10 @@ void fsl_ddr_set_memctl_regs(const fsl_ddr_cfg_regs_t *regs,
        u32 vref_seq2[3] = {0xc0, 0xf0, 0x70};  /* for range 2 */
        u32 *vref_seq = vref_seq1;
 #endif
+#ifdef CONFIG_SYS_FSL_ERRATUM_A009942
+       ulong ddr_freq;
+       u32 tmp;
+#endif
 #ifdef CONFIG_FSL_DDR_BIST
        u32 mtcr, err_detect, err_sbe;
        u32 cs0_bnds, cs1_bnds, cs2_bnds, cs3_bnds, cs0_config;
@@ -227,6 +231,20 @@ void fsl_ddr_set_memctl_regs(const fsl_ddr_cfg_regs_t *regs,
                ddr_out32(&ddr->debug[25], 0x9000);
        }
 #endif
+
+#ifdef CONFIG_SYS_FSL_ERRATUM_A009942
+       ddr_freq = get_ddr_freq(ctrl_num) / 1000000;
+       tmp = ddr_in32(&ddr->debug[28]);
+       if (ddr_freq <= 1333)
+               ddr_out32(&ddr->debug[28], tmp | 0x0080006a);
+       else if (ddr_freq <= 1600)
+               ddr_out32(&ddr->debug[28], tmp | 0x0070006f);
+       else if (ddr_freq <= 1867)
+               ddr_out32(&ddr->debug[28], tmp | 0x00700076);
+       else if (ddr_freq <= 2133)
+               ddr_out32(&ddr->debug[28], tmp | 0x0060007b);
+#endif
+
        /*
         * For RDIMMs, JEDEC spec requires clocks to be stable before reset is
         * deasserted. Clocks start when any chip select is enabled and clock