1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26#include <common.h>
27
28#include <asm/errno.h>
29#include <linux/mtd/mtd.h>
30#include <linux/mtd/nand_ecc.h>
31
32
33#ifdef CONFIG_NAND_NDFC
34#define CONFIG_MTD_NAND_ECC_SMC
35#endif
36
37
38
39
40
41
42#if !defined(CONFIG_NAND_SPL) || defined(CONFIG_SPL_NAND_SOFTECC)
43
44
45
46static const u_char nand_ecc_precalc_table[] = {
47 0x00, 0x55, 0x56, 0x03, 0x59, 0x0c, 0x0f, 0x5a, 0x5a, 0x0f, 0x0c, 0x59, 0x03, 0x56, 0x55, 0x00,
48 0x65, 0x30, 0x33, 0x66, 0x3c, 0x69, 0x6a, 0x3f, 0x3f, 0x6a, 0x69, 0x3c, 0x66, 0x33, 0x30, 0x65,
49 0x66, 0x33, 0x30, 0x65, 0x3f, 0x6a, 0x69, 0x3c, 0x3c, 0x69, 0x6a, 0x3f, 0x65, 0x30, 0x33, 0x66,
50 0x03, 0x56, 0x55, 0x00, 0x5a, 0x0f, 0x0c, 0x59, 0x59, 0x0c, 0x0f, 0x5a, 0x00, 0x55, 0x56, 0x03,
51 0x69, 0x3c, 0x3f, 0x6a, 0x30, 0x65, 0x66, 0x33, 0x33, 0x66, 0x65, 0x30, 0x6a, 0x3f, 0x3c, 0x69,
52 0x0c, 0x59, 0x5a, 0x0f, 0x55, 0x00, 0x03, 0x56, 0x56, 0x03, 0x00, 0x55, 0x0f, 0x5a, 0x59, 0x0c,
53 0x0f, 0x5a, 0x59, 0x0c, 0x56, 0x03, 0x00, 0x55, 0x55, 0x00, 0x03, 0x56, 0x0c, 0x59, 0x5a, 0x0f,
54 0x6a, 0x3f, 0x3c, 0x69, 0x33, 0x66, 0x65, 0x30, 0x30, 0x65, 0x66, 0x33, 0x69, 0x3c, 0x3f, 0x6a,
55 0x6a, 0x3f, 0x3c, 0x69, 0x33, 0x66, 0x65, 0x30, 0x30, 0x65, 0x66, 0x33, 0x69, 0x3c, 0x3f, 0x6a,
56 0x0f, 0x5a, 0x59, 0x0c, 0x56, 0x03, 0x00, 0x55, 0x55, 0x00, 0x03, 0x56, 0x0c, 0x59, 0x5a, 0x0f,
57 0x0c, 0x59, 0x5a, 0x0f, 0x55, 0x00, 0x03, 0x56, 0x56, 0x03, 0x00, 0x55, 0x0f, 0x5a, 0x59, 0x0c,
58 0x69, 0x3c, 0x3f, 0x6a, 0x30, 0x65, 0x66, 0x33, 0x33, 0x66, 0x65, 0x30, 0x6a, 0x3f, 0x3c, 0x69,
59 0x03, 0x56, 0x55, 0x00, 0x5a, 0x0f, 0x0c, 0x59, 0x59, 0x0c, 0x0f, 0x5a, 0x00, 0x55, 0x56, 0x03,
60 0x66, 0x33, 0x30, 0x65, 0x3f, 0x6a, 0x69, 0x3c, 0x3c, 0x69, 0x6a, 0x3f, 0x65, 0x30, 0x33, 0x66,
61 0x65, 0x30, 0x33, 0x66, 0x3c, 0x69, 0x6a, 0x3f, 0x3f, 0x6a, 0x69, 0x3c, 0x66, 0x33, 0x30, 0x65,
62 0x00, 0x55, 0x56, 0x03, 0x59, 0x0c, 0x0f, 0x5a, 0x5a, 0x0f, 0x0c, 0x59, 0x03, 0x56, 0x55, 0x00
63};
64
65
66
67
68
69
70
71int nand_calculate_ecc(struct mtd_info *mtd, const u_char *dat,
72 u_char *ecc_code)
73{
74 uint8_t idx, reg1, reg2, reg3, tmp1, tmp2;
75 int i;
76
77
78 reg1 = reg2 = reg3 = 0;
79
80
81 for(i = 0; i < 256; i++) {
82
83 idx = nand_ecc_precalc_table[*dat++];
84 reg1 ^= (idx & 0x3f);
85
86
87 if (idx & 0x40) {
88 reg3 ^= (uint8_t) i;
89 reg2 ^= ~((uint8_t) i);
90 }
91 }
92
93
94 tmp1 = (reg3 & 0x80) >> 0;
95 tmp1 |= (reg2 & 0x80) >> 1;
96 tmp1 |= (reg3 & 0x40) >> 1;
97 tmp1 |= (reg2 & 0x40) >> 2;
98 tmp1 |= (reg3 & 0x20) >> 2;
99 tmp1 |= (reg2 & 0x20) >> 3;
100 tmp1 |= (reg3 & 0x10) >> 3;
101 tmp1 |= (reg2 & 0x10) >> 4;
102
103 tmp2 = (reg3 & 0x08) << 4;
104 tmp2 |= (reg2 & 0x08) << 3;
105 tmp2 |= (reg3 & 0x04) << 3;
106 tmp2 |= (reg2 & 0x04) << 2;
107 tmp2 |= (reg3 & 0x02) << 2;
108 tmp2 |= (reg2 & 0x02) << 1;
109 tmp2 |= (reg3 & 0x01) << 1;
110 tmp2 |= (reg2 & 0x01) << 0;
111
112
113#ifdef CONFIG_MTD_NAND_ECC_SMC
114 ecc_code[0] = ~tmp2;
115 ecc_code[1] = ~tmp1;
116#else
117 ecc_code[0] = ~tmp1;
118 ecc_code[1] = ~tmp2;
119#endif
120 ecc_code[2] = ((~reg1) << 2) | 0x03;
121
122 return 0;
123}
124#endif
125
126static inline int countbits(uint32_t byte)
127{
128 int res = 0;
129
130 for (;byte; byte >>= 1)
131 res += byte & 0x01;
132 return res;
133}
134
135
136
137
138
139
140
141
142
143
144int nand_correct_data(struct mtd_info *mtd, u_char *dat,
145 u_char *read_ecc, u_char *calc_ecc)
146{
147 uint8_t s0, s1, s2;
148
149#ifdef CONFIG_MTD_NAND_ECC_SMC
150 s0 = calc_ecc[0] ^ read_ecc[0];
151 s1 = calc_ecc[1] ^ read_ecc[1];
152 s2 = calc_ecc[2] ^ read_ecc[2];
153#else
154 s1 = calc_ecc[0] ^ read_ecc[0];
155 s0 = calc_ecc[1] ^ read_ecc[1];
156 s2 = calc_ecc[2] ^ read_ecc[2];
157#endif
158 if ((s0 | s1 | s2) == 0)
159 return 0;
160
161
162 if( ((s0 ^ (s0 >> 1)) & 0x55) == 0x55 &&
163 ((s1 ^ (s1 >> 1)) & 0x55) == 0x55 &&
164 ((s2 ^ (s2 >> 1)) & 0x54) == 0x54) {
165
166 uint32_t byteoffs, bitnum;
167
168 byteoffs = (s1 << 0) & 0x80;
169 byteoffs |= (s1 << 1) & 0x40;
170 byteoffs |= (s1 << 2) & 0x20;
171 byteoffs |= (s1 << 3) & 0x10;
172
173 byteoffs |= (s0 >> 4) & 0x08;
174 byteoffs |= (s0 >> 3) & 0x04;
175 byteoffs |= (s0 >> 2) & 0x02;
176 byteoffs |= (s0 >> 1) & 0x01;
177
178 bitnum = (s2 >> 5) & 0x04;
179 bitnum |= (s2 >> 4) & 0x02;
180 bitnum |= (s2 >> 3) & 0x01;
181
182 dat[byteoffs] ^= (1 << bitnum);
183
184 return 1;
185 }
186
187 if(countbits(s0 | ((uint32_t)s1 << 8) | ((uint32_t)s2 <<16)) == 1)
188 return 1;
189
190 return -EBADMSG;
191}
192