blob: 88a4605ad5d34f80913038160c05f60fccc66736 [file] [log] [blame]
Grant Ericksonb6933412008-05-22 14:44:14 -07001/*
2 * Copyright (c) 2008 Nuovation System Designs, LLC
3 * Grant Erickson <gerickson@nuovations.com>
4 *
Stefan Roesecb9ebd02009-09-28 17:33:45 +02005 * (C) Copyright 2005-2009
Grant Ericksonb6933412008-05-22 14:44:14 -07006 * Stefan Roese, DENX Software Engineering, sr@denx.de.
7 *
8 * (C) Copyright 2002
9 * Jun Gu, Artesyn Technology, jung@artesyncp.com
10 *
11 * (C) Copyright 2001
12 * Bill Hunter, Wave 7 Optics, williamhunter@attbi.com
13 *
Wolfgang Denkd79de1d2013-07-08 09:37:19 +020014 * SPDX-License-Identifier: GPL-2.0+
Grant Ericksonb6933412008-05-22 14:44:14 -070015 *
16 * Description:
17 * This file implements generic DRAM ECC initialization for
18 * PowerPC processors using a SDRAM DDR/DDR2 controller,
19 * including the 405EX(r), 440GP/GX/EP/GR, 440SP(E), and
20 * 460EX/GT.
21 */
22
23#include <common.h>
Stefan Roese247e9d72010-09-09 19:18:00 +020024#include <asm/ppc4xx.h>
Grant Ericksonb6933412008-05-22 14:44:14 -070025#include <ppc_asm.tmpl>
26#include <ppc_defs.h>
27#include <asm/processor.h>
28#include <asm/io.h>
Stefan Roesecb9ebd02009-09-28 17:33:45 +020029#include <asm/mmu.h>
30#include <asm/cache.h>
Grant Ericksonb6933412008-05-22 14:44:14 -070031
32#include "ecc.h"
33
Stefan Roesedb7d0ab2008-06-02 17:22:11 +020034#if defined(CONFIG_SDRAM_PPC4xx_IBM_DDR) || \
35 defined(CONFIG_SDRAM_PPC4xx_IBM_DDR2)
Grant Ericksonb6933412008-05-22 14:44:14 -070036#if defined(CONFIG_DDR_ECC) || defined(CONFIG_SDRAM_ECC)
Stefan Roesecb9ebd02009-09-28 17:33:45 +020037
38#if defined(CONFIG_405EX)
Grant Ericksonb6933412008-05-22 14:44:14 -070039/*
Stefan Roesecb9ebd02009-09-28 17:33:45 +020040 * Currently only 405EX uses 16bit data bus width as an alternative
41 * option to 32bit data width (SDRAM0_MCOPT1_WDTH)
Grant Ericksonb6933412008-05-22 14:44:14 -070042 */
Stefan Roesecb9ebd02009-09-28 17:33:45 +020043#define SDRAM_DATA_ALT_WIDTH 2
44#else
45#define SDRAM_DATA_ALT_WIDTH 8
46#endif
47
48static void wait_ddr_idle(void)
Grant Ericksonb6933412008-05-22 14:44:14 -070049{
Stefan Roesecb9ebd02009-09-28 17:33:45 +020050 u32 val;
51
52 do {
53 mfsdram(SDRAM_MCSTAT, val);
54 } while ((val & SDRAM_MCSTAT_IDLE_MASK) == SDRAM_MCSTAT_IDLE_NOT);
55}
56
57static void program_ecc_addr(unsigned long start_address,
58 unsigned long num_bytes,
59 unsigned long tlb_word2_i_value)
60{
61 unsigned long current_address;
62 unsigned long end_address;
63 unsigned long address_increment;
Grant Ericksonb6933412008-05-22 14:44:14 -070064 unsigned long mcopt1;
Stefan Roesecb9ebd02009-09-28 17:33:45 +020065 char str[] = "ECC generation -";
66 char slash[] = "\\|/-\\|/-";
67 int loop = 0;
68 int loopi = 0;
Grant Ericksonb6933412008-05-22 14:44:14 -070069
Stefan Roesecb9ebd02009-09-28 17:33:45 +020070 current_address = start_address;
71 mfsdram(SDRAM_MCOPT1, mcopt1);
72 if ((mcopt1 & SDRAM_MCOPT1_MCHK_MASK) != SDRAM_MCOPT1_MCHK_NON) {
73 mtsdram(SDRAM_MCOPT1,
74 (mcopt1 & ~SDRAM_MCOPT1_MCHK_MASK) | SDRAM_MCOPT1_MCHK_GEN);
75 sync();
76 eieio();
77 wait_ddr_idle();
78
79 puts(str);
Grant Ericksonb6933412008-05-22 14:44:14 -070080
Stefan Roesecb9ebd02009-09-28 17:33:45 +020081#ifdef CONFIG_440
82 if (tlb_word2_i_value == TLB_WORD2_I_ENABLE) {
83#endif
84 /* ECC bit set method for non-cached memory */
85 if ((mcopt1 & SDRAM_MCOPT1_DMWD_MASK) == SDRAM_MCOPT1_DMWD_32)
86 address_increment = 4;
87 else
88 address_increment = SDRAM_DATA_ALT_WIDTH;
89 end_address = current_address + num_bytes;
Grant Ericksonb6933412008-05-22 14:44:14 -070090
Stefan Roesecb9ebd02009-09-28 17:33:45 +020091 while (current_address < end_address) {
92 *((unsigned long *)current_address) = 0;
93 current_address += address_increment;
Grant Ericksonb6933412008-05-22 14:44:14 -070094
Stefan Roesecb9ebd02009-09-28 17:33:45 +020095 if ((loop++ % (2 << 20)) == 0) {
96 putc('\b');
97 putc(slash[loopi++ % 8]);
98 }
99 }
100#ifdef CONFIG_440
101 } else {
102 /* ECC bit set method for cached memory */
103 dcbz_area(start_address, num_bytes);
104 /* Write modified dcache lines back to memory */
105 clean_dcache_range(start_address, start_address + num_bytes);
106 }
107#endif /* CONFIG_440 */
Grant Ericksonb6933412008-05-22 14:44:14 -0700108
Stefan Roesecb9ebd02009-09-28 17:33:45 +0200109 blank_string(strlen(str));
Grant Ericksonb6933412008-05-22 14:44:14 -0700110
Stefan Roesecb9ebd02009-09-28 17:33:45 +0200111 sync();
112 eieio();
113 wait_ddr_idle();
114
115 /* clear ECC error repoting registers */
116 mtsdram(SDRAM_ECCES, 0xffffffff);
Stefan Roese971edca2010-07-21 11:08:27 +0200117#if defined(CONFIG_SDRAM_PPC4xx_IBM_DDR)
118 /*
119 * IBM DDR(1) core (440GX):
120 * Clear Mx bits in SDRAM0_BESR0/1
121 */
122 mtsdram(SDRAM0_BESR0, 0xffffffff);
123 mtsdram(SDRAM0_BESR1, 0xffffffff);
124#elif defined(CONFIG_440)
125 /*
126 * 440/460 DDR2 core:
127 * Clear EMID (Error PLB Master ID) in MQ0_ESL
128 */
129 mtdcr(SDRAM_ERRSTATLL, 0xfff00000);
130#else
131 /*
132 * 405EX(r) DDR2 core:
133 * Clear M0ID (Error PLB Master ID) in SDRAM_BESR
134 */
135 mtsdram(SDRAM_BESR, 0xf0000000);
136#endif
Stefan Roesecb9ebd02009-09-28 17:33:45 +0200137
138 mtsdram(SDRAM_MCOPT1,
139 (mcopt1 & ~SDRAM_MCOPT1_MCHK_MASK) | SDRAM_MCOPT1_MCHK_CHK_REP);
140 sync();
141 eieio();
142 wait_ddr_idle();
143 }
144}
145
146#if defined(CONFIG_SDRAM_PPC4xx_IBM_DDR)
147void ecc_init(unsigned long * const start, unsigned long size)
148{
Grant Ericksonb6933412008-05-22 14:44:14 -0700149 /*
Stefan Roesecb9ebd02009-09-28 17:33:45 +0200150 * Init ECC with cache disabled (on PPC's with IBM DDR
151 * controller (non DDR2), not tested with cache enabled yet
Grant Ericksonb6933412008-05-22 14:44:14 -0700152 */
Stefan Roesecb9ebd02009-09-28 17:33:45 +0200153 program_ecc_addr((u32)start, size, TLB_WORD2_I_ENABLE);
154}
155#endif
Grant Ericksonb6933412008-05-22 14:44:14 -0700156
Stefan Roesecb9ebd02009-09-28 17:33:45 +0200157#if defined(CONFIG_SDRAM_PPC4xx_IBM_DDR2)
158void do_program_ecc(unsigned long tlb_word2_i_value)
159{
160 unsigned long mcopt1;
161 unsigned long mcopt2;
162 unsigned long mcstat;
163 phys_size_t memsize = sdram_memsize();
Grant Ericksonb6933412008-05-22 14:44:14 -0700164
Stefan Roesecb9ebd02009-09-28 17:33:45 +0200165 if (memsize > CONFIG_MAX_MEM_MAPPED) {
166 printf("\nWarning: Can't enable ECC on systems with more than 2GB of SDRAM!\n");
167 return;
Grant Ericksonb6933412008-05-22 14:44:14 -0700168 }
169
Stefan Roesecb9ebd02009-09-28 17:33:45 +0200170 mfsdram(SDRAM_MCOPT1, mcopt1);
171 mfsdram(SDRAM_MCOPT2, mcopt2);
Grant Ericksonb6933412008-05-22 14:44:14 -0700172
Stefan Roesecb9ebd02009-09-28 17:33:45 +0200173 if ((mcopt1 & SDRAM_MCOPT1_MCHK_MASK) != SDRAM_MCOPT1_MCHK_NON) {
174 /* DDR controller must be enabled and not in self-refresh. */
175 mfsdram(SDRAM_MCSTAT, mcstat);
176 if (((mcopt2 & SDRAM_MCOPT2_DCEN_MASK) == SDRAM_MCOPT2_DCEN_ENABLE)
177 && ((mcopt2 & SDRAM_MCOPT2_SREN_MASK) == SDRAM_MCOPT2_SREN_EXIT)
178 && ((mcstat & (SDRAM_MCSTAT_MIC_MASK | SDRAM_MCSTAT_SRMS_MASK))
179 == (SDRAM_MCSTAT_MIC_COMP | SDRAM_MCSTAT_SRMS_NOT_SF))) {
Grant Ericksonb6933412008-05-22 14:44:14 -0700180
Stefan Roesecb9ebd02009-09-28 17:33:45 +0200181 program_ecc_addr(0, memsize, tlb_word2_i_value);
182 }
183 }
Grant Ericksonb6933412008-05-22 14:44:14 -0700184}
Stefan Roesecb9ebd02009-09-28 17:33:45 +0200185#endif
186
Grant Ericksonb6933412008-05-22 14:44:14 -0700187#endif /* defined(CONFIG_DDR_ECC) || defined(CONFIG_SDRAM_ECC) */
Stefan Roesedb7d0ab2008-06-02 17:22:11 +0200188#endif /* defined(CONFIG_SDRAM_PPC4xx_IBM_DDR)... */