xref: /netbsd-src/sys/dev/nand/hamming.c (revision f273a7a174ebed441f3363e0c944bd42390ca256)
1 /*	$NetBSD: hamming.c,v 1.2 2021/12/07 21:37:37 andvar Exp $	*/
2 
3 /*
4  * Copyright (c) 2008, Atmel Corporation
5  *
6  * All rights reserved.
7  *
8  * Redistribution and use in source and binary forms, with or without
9  * modification, are permitted provided that the following conditions are met:
10  *
11  * - Redistributions of source code must retain the above copyright notice,
12  * this list of conditions and the disclaimer below.
13  *
14  * Atmel's name may not be used to endorse or promote products derived from
15  * this software without specific prior written permission.
16  *
17  * DISCLAIMER: THIS SOFTWARE IS PROVIDED BY ATMEL "AS IS" AND ANY EXPRESS OR
18  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
19  * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT ARE
20  * DISCLAIMED. IN NO EVENT SHALL ATMEL BE LIABLE FOR ANY DIRECT, INDIRECT,
21  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
22  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
23  * OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
24  * LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
25  * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
26  * EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27  */
28 
29 #include <sys/cdefs.h>
30 __KERNEL_RCSID(0, "$NetBSD: hamming.c,v 1.2 2021/12/07 21:37:37 andvar Exp $");
31 
32 #include <sys/param.h>
33 #include <lib/libkern/libkern.h>
34 #include "hamming.h"
35 
36 /**
37  * Calculates the 22-bit hamming code for a 256-bytes block of data.
38  * \param data  Data buffer to calculate code for.
39  * \param code  Pointer to a buffer where the code should be stored.
40  */
41 void
hamming_compute_256(const uint8_t * data,uint8_t * code)42 hamming_compute_256(const uint8_t *data, uint8_t *code)
43 {
44 	unsigned int i;
45 	uint8_t column_sum = 0;
46 	uint8_t even_line_code = 0;
47 	uint8_t odd_line_code = 0;
48 	uint8_t even_column_code = 0;
49 	uint8_t odd_column_code = 0;
50 
51 	/*-
52 	 * Xor all bytes together to get the column sum;
53 	 * At the same time, calculate the even and odd line codes
54 	 */
55 	for (i = 0; i < 256; i++) {
56 		column_sum ^= data[i];
57 
58 		/*-
59 		 * If the xor sum of the byte is 0, then this byte has no
60 		 * incidence on the computed code; so check if the sum is 1.
61 		 */
62 		if ((popcount(data[i]) & 1) == 1) {
63 			/*-
64 			 * Parity groups are formed by forcing a particular
65 			 * index bit to 0 (even) or 1 (odd).
66 			 * Example on one byte:
67 			 *
68 			 * bits (dec)  7   6   5   4   3   2   1   0
69 			 *      (bin) 111 110 101 100 011 010 001 000
70 			 *                            '---'---'---'----------.
71 			 *                                                   |
72 			 * groups P4' ooooooooooooooo eeeeeeeeeeeeeee P4     |
73 			 *        P2' ooooooo eeeeeee ooooooo eeeeeee P2     |
74 			 *        P1' ooo eee ooo eee ooo eee ooo eee P1     |
75 			 *                                                   |
76 			 * We can see that:                                  |
77 			 *  - P4  -> bit 2 of index is 0 --------------------'
78 			 *  - P4' -> bit 2 of index is 1.
79 			 *  - P2  -> bit 1 of index if 0.
80 			 *  - etc...
81 			 * We deduce that a bit position has an impact on all
82 			 * even Px if the log2(x)nth bit of its index is 0
83 			 *     ex: log2(4) = 2,
84 			 * bit2 of the index must be 0 (-> 0 1 2 3)
85 			 * and on all odd Px' if the log2(x)nth bit
86 			 * of its index is 1
87 			 *     ex: log2(2) = 1,
88 			 * bit1 of the index must be 1 (-> 0 1 4 5)
89 			 *
90 			 * As such, we calculate all the possible Px and Px'
91 			 * values at the same time in two variables,
92 			 * even_line_code and odd_line_code, such as
93 			 *     even_line_code bits: P128  P64  P32
94 			 *                        P16  P8  P4  P2  P1
95 			 *     odd_line_code  bits: P128' P64' P32' P16'
96 			 *                        P8' P4' P2' P1'
97 			 */
98 			even_line_code ^= (255 - i);
99 			odd_line_code ^= i;
100 		}
101 	}
102 
103 	/*-
104 	 * At this point, we have the line parities, and the column sum.
105 	 * First, We must calculate the parity group values on the column sum.
106 	 */
107 	for (i = 0; i < 8; i++) {
108 		if (column_sum & 1) {
109 			even_column_code ^= (7 - i);
110 			odd_column_code ^= i;
111 		}
112 		column_sum >>= 1;
113 	}
114 
115 	/*-
116 	 * Now, we must interleave the parity values,
117 	 * to obtain the following layout:
118 	 * Code[0] = Line1
119 	 * Code[1] = Line2
120 	 * Code[2] = Column
121 	 * Line = Px' Px P(x-1)- P(x-1) ...
122 	 * Column = P4' P4 P2' P2 P1' P1 PadBit PadBit
123 	 */
124 	code[0] = 0;
125 	code[1] = 0;
126 	code[2] = 0;
127 
128 	for (i = 0; i < 4; i++) {
129 		code[0] <<= 2;
130 		code[1] <<= 2;
131 		code[2] <<= 2;
132 
133 		/* Line 1 */
134 		if ((odd_line_code & 0x80) != 0) {
135 
136 			code[0] |= 2;
137 		}
138 		if ((even_line_code & 0x80) != 0) {
139 
140 			code[0] |= 1;
141 		}
142 
143 		/* Line 2 */
144 		if ((odd_line_code & 0x08) != 0) {
145 
146 			code[1] |= 2;
147 		}
148 		if ((even_line_code & 0x08) != 0) {
149 
150 			code[1] |= 1;
151 		}
152 
153 		/* Column */
154 		if ((odd_column_code & 0x04) != 0) {
155 
156 			code[2] |= 2;
157 		}
158 		if ((even_column_code & 0x04) != 0) {
159 
160 			code[2] |= 1;
161 		}
162 
163 		odd_line_code <<= 1;
164 		even_line_code <<= 1;
165 		odd_column_code <<= 1;
166 		even_column_code <<= 1;
167 	}
168 
169 	/* Invert codes (linux compatibility) */
170 	code[0] = ~code[0];
171 	code[1] = ~code[1];
172 	code[2] = ~code[2];
173 }
174 
175 /**
176  * Verifies and corrects a 256-bytes block of data using the given 22-bits
177  * hamming code.
178  * Returns 0 if there is no error, otherwise returns a HAMMING_ERROR code.
179  * param data  Data buffer to check.
180  * \param original_code  Hamming code to use for verifying the data.
181  */
182 uint8_t
hamming_correct_256(uint8_t * data,const uint8_t * original_code,const uint8_t * computed_code)183 hamming_correct_256(uint8_t *data, const uint8_t *original_code,
184     const uint8_t *computed_code)
185 {
186 	/* Calculate new code */
187 	/* we allocate 4 bytes so we can use popcount32 in one step */
188 	uint8_t correction_code[4];
189 
190 	/* this byte should remain zero all the time */
191 	correction_code[3] = 0;
192 
193 	/* Xor both codes together */
194 	correction_code[0] = computed_code[0] ^ original_code[0];
195 	correction_code[1] = computed_code[1] ^ original_code[1];
196 	correction_code[2] = computed_code[2] ^ original_code[2];
197 
198 	/* If all bytes are 0, there is no error */
199 	if (*(uint32_t *)correction_code == 0) {
200 		return 0;
201 	}
202 	/* If there is a single bit error, there are 11 bits set to 1 */
203 	if (popcount32(*(uint32_t *)correction_code) == 11) {
204 		/* Get byte and bit indexes */
205 		uint8_t byte = correction_code[0] & 0x80;
206 		byte |= (correction_code[0] << 1) & 0x40;
207 		byte |= (correction_code[0] << 2) & 0x20;
208 		byte |= (correction_code[0] << 3) & 0x10;
209 
210 		byte |= (correction_code[1] >> 4) & 0x08;
211 		byte |= (correction_code[1] >> 3) & 0x04;
212 		byte |= (correction_code[1] >> 2) & 0x02;
213 		byte |= (correction_code[1] >> 1) & 0x01;
214 
215 		uint8_t bit = (correction_code[2] >> 5) & 0x04;
216 		bit |= (correction_code[2] >> 4) & 0x02;
217 		bit |= (correction_code[2] >> 3) & 0x01;
218 
219 		/* Correct bit */
220 		data[byte] ^= (1 << bit);
221 
222 		return HAMMING_ERROR_SINGLEBIT;
223 	}
224 	/* Check if ECC has been corrupted */
225 	if (popcount32(*(uint32_t *)correction_code) == 1) {
226 		return HAMMING_ERROR_ECC;
227 	} else {
228 		/* Otherwise, this is a multi-bit error */
229 		return HAMMING_ERROR_MULTIPLEBITS;
230 	}
231 }
232 
233