Blame view

fs/ext4/hash.c 6.12 KB
f51667685   Theodore Ts'o   ext4: fix up rema...
1
  // SPDX-License-Identifier: GPL-2.0
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
2
  /*
617ba13b3   Mingming Cao   [PATCH] ext4: ren...
3
   *  linux/fs/ext4/hash.c
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
4
5
   *
   * Copyright (C) 2002 by Theodore Ts'o
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
6
7
8
   */
  
  #include <linux/fs.h>
1c83a9aab   Jason A. Donenfeld   ext4: move halfmd...
9
10
  #include <linux/compiler.h>
  #include <linux/bitops.h>
3dcf54515   Christoph Hellwig   ext4: move header...
11
  #include "ext4.h"
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
12
13
14
15
16
17
18
19
20
21
22
23
24
25
  
  #define DELTA 0x9E3779B9
  
  static void TEA_transform(__u32 buf[4], __u32 const in[])
  {
  	__u32	sum = 0;
  	__u32	b0 = buf[0], b1 = buf[1];
  	__u32	a = in[0], b = in[1], c = in[2], d = in[3];
  	int	n = 16;
  
  	do {
  		sum += DELTA;
  		b0 += ((b1 << 4)+a) ^ (b1+sum) ^ ((b1 >> 5)+b);
  		b1 += ((b0 << 4)+c) ^ (b0+sum) ^ ((b0 >> 5)+d);
af5bc92dd   Theodore Ts'o   ext4: Fix whitesp...
26
  	} while (--n);
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
27
28
29
30
  
  	buf[0] += b0;
  	buf[1] += b1;
  }
1c83a9aab   Jason A. Donenfeld   ext4: move halfmd...
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
  /* F, G and H are basic MD4 functions: selection, majority, parity */
  #define F(x, y, z) ((z) ^ ((x) & ((y) ^ (z))))
  #define G(x, y, z) (((x) & (y)) + (((x) ^ (y)) & (z)))
  #define H(x, y, z) ((x) ^ (y) ^ (z))
  
  /*
   * The generic round function.  The application is so specific that
   * we don't bother protecting all the arguments with parens, as is generally
   * good macro practice, in favor of extra legibility.
   * Rotation is separate from addition to prevent recomputation
   */
  #define ROUND(f, a, b, c, d, x, s)	\
  	(a += f(b, c, d) + x, a = rol32(a, s))
  #define K1 0
  #define K2 013240474631UL
  #define K3 015666365641UL
  
  /*
   * Basic cut-down MD4 transform.  Returns only 32 bits of result.
   */
  static __u32 half_md4_transform(__u32 buf[4], __u32 const in[8])
  {
  	__u32 a = buf[0], b = buf[1], c = buf[2], d = buf[3];
  
  	/* Round 1 */
  	ROUND(F, a, b, c, d, in[0] + K1,  3);
  	ROUND(F, d, a, b, c, in[1] + K1,  7);
  	ROUND(F, c, d, a, b, in[2] + K1, 11);
  	ROUND(F, b, c, d, a, in[3] + K1, 19);
  	ROUND(F, a, b, c, d, in[4] + K1,  3);
  	ROUND(F, d, a, b, c, in[5] + K1,  7);
  	ROUND(F, c, d, a, b, in[6] + K1, 11);
  	ROUND(F, b, c, d, a, in[7] + K1, 19);
  
  	/* Round 2 */
  	ROUND(G, a, b, c, d, in[1] + K2,  3);
  	ROUND(G, d, a, b, c, in[3] + K2,  5);
  	ROUND(G, c, d, a, b, in[5] + K2,  9);
  	ROUND(G, b, c, d, a, in[7] + K2, 13);
  	ROUND(G, a, b, c, d, in[0] + K2,  3);
  	ROUND(G, d, a, b, c, in[2] + K2,  5);
  	ROUND(G, c, d, a, b, in[4] + K2,  9);
  	ROUND(G, b, c, d, a, in[6] + K2, 13);
  
  	/* Round 3 */
  	ROUND(H, a, b, c, d, in[3] + K3,  3);
  	ROUND(H, d, a, b, c, in[7] + K3,  9);
  	ROUND(H, c, d, a, b, in[2] + K3, 11);
  	ROUND(H, b, c, d, a, in[6] + K3, 15);
  	ROUND(H, a, b, c, d, in[1] + K3,  3);
  	ROUND(H, d, a, b, c, in[5] + K3,  9);
  	ROUND(H, c, d, a, b, in[0] + K3, 11);
  	ROUND(H, b, c, d, a, in[4] + K3, 15);
  
  	buf[0] += a;
  	buf[1] += b;
  	buf[2] += c;
  	buf[3] += d;
  
  	return buf[1]; /* "most hashed" word */
  }
  #undef ROUND
  #undef K1
  #undef K2
  #undef K3
  #undef F
  #undef G
  #undef H
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
99
100
  
  /* The old legacy hash */
f99b25897   Theodore Ts'o   ext4: Add support...
101
  static __u32 dx_hack_hash_unsigned(const char *name, int len)
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
102
  {
f99b25897   Theodore Ts'o   ext4: Add support...
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
  	__u32 hash, hash0 = 0x12a3fe2d, hash1 = 0x37abe8f9;
  	const unsigned char *ucp = (const unsigned char *) name;
  
  	while (len--) {
  		hash = hash1 + (hash0 ^ (((int) *ucp++) * 7152373));
  
  		if (hash & 0x80000000)
  			hash -= 0x7fffffff;
  		hash1 = hash0;
  		hash0 = hash;
  	}
  	return hash0 << 1;
  }
  
  static __u32 dx_hack_hash_signed(const char *name, int len)
  {
  	__u32 hash, hash0 = 0x12a3fe2d, hash1 = 0x37abe8f9;
  	const signed char *scp = (const signed char *) name;
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
121
  	while (len--) {
f99b25897   Theodore Ts'o   ext4: Add support...
122
  		hash = hash1 + (hash0 ^ (((int) *scp++) * 7152373));
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
123

f99b25897   Theodore Ts'o   ext4: Add support...
124
125
  		if (hash & 0x80000000)
  			hash -= 0x7fffffff;
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
126
127
128
  		hash1 = hash0;
  		hash0 = hash;
  	}
f99b25897   Theodore Ts'o   ext4: Add support...
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
  	return hash0 << 1;
  }
  
  static void str2hashbuf_signed(const char *msg, int len, __u32 *buf, int num)
  {
  	__u32	pad, val;
  	int	i;
  	const signed char *scp = (const signed char *) msg;
  
  	pad = (__u32)len | ((__u32)len << 8);
  	pad |= pad << 16;
  
  	val = pad;
  	if (len > num*4)
  		len = num * 4;
  	for (i = 0; i < len; i++) {
f99b25897   Theodore Ts'o   ext4: Add support...
145
146
147
148
149
150
151
152
153
154
155
  		val = ((int) scp[i]) + (val << 8);
  		if ((i % 4) == 3) {
  			*buf++ = val;
  			val = pad;
  			num--;
  		}
  	}
  	if (--num >= 0)
  		*buf++ = val;
  	while (--num >= 0)
  		*buf++ = pad;
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
156
  }
f99b25897   Theodore Ts'o   ext4: Add support...
157
  static void str2hashbuf_unsigned(const char *msg, int len, __u32 *buf, int num)
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
158
159
160
  {
  	__u32	pad, val;
  	int	i;
f99b25897   Theodore Ts'o   ext4: Add support...
161
  	const unsigned char *ucp = (const unsigned char *) msg;
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
162
163
164
165
166
167
168
  
  	pad = (__u32)len | ((__u32)len << 8);
  	pad |= pad << 16;
  
  	val = pad;
  	if (len > num*4)
  		len = num * 4;
af5bc92dd   Theodore Ts'o   ext4: Fix whitesp...
169
  	for (i = 0; i < len; i++) {
f99b25897   Theodore Ts'o   ext4: Add support...
170
  		val = ((int) ucp[i]) + (val << 8);
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
  		if ((i % 4) == 3) {
  			*buf++ = val;
  			val = pad;
  			num--;
  		}
  	}
  	if (--num >= 0)
  		*buf++ = val;
  	while (--num >= 0)
  		*buf++ = pad;
  }
  
  /*
   * Returns the hash of a filename.  If len is 0 and name is NULL, then
   * this function can be used to test whether or not a hash version is
   * supported.
   *
   * The seed is an 4 longword (32 bits) "secret" which can be used to
   * uniquify a hash.  If the seed is all zero's, then some default seed
   * may be used.
   *
   * A particular hash version specifies whether or not the seed is
   * represented, and whether or not the returned hash is 32 bits or 64
   * bits.  32 bit hashes will return 0 for the minor hash.
   */
617ba13b3   Mingming Cao   [PATCH] ext4: ren...
196
  int ext4fs_dirhash(const char *name, int len, struct dx_hash_info *hinfo)
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
197
198
199
200
201
202
  {
  	__u32	hash;
  	__u32	minor_hash = 0;
  	const char	*p;
  	int		i;
  	__u32		in[8], buf[4];
f99b25897   Theodore Ts'o   ext4: Add support...
203
204
  	void		(*str2hashbuf)(const char *, int, __u32 *, int) =
  				str2hashbuf_signed;
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
205
206
207
208
209
210
211
212
213
  
  	/* Initialize the default seed for the hash checksum functions */
  	buf[0] = 0x67452301;
  	buf[1] = 0xefcdab89;
  	buf[2] = 0x98badcfe;
  	buf[3] = 0x10325476;
  
  	/* Check to see if the seed is all zero's */
  	if (hinfo->seed) {
af5bc92dd   Theodore Ts'o   ext4: Fix whitesp...
214
  		for (i = 0; i < 4; i++) {
0e79537d3   Cong Ding   ext4: reduce one ...
215
216
  			if (hinfo->seed[i]) {
  				memcpy(buf, hinfo->seed, sizeof(buf));
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
217
  				break;
0e79537d3   Cong Ding   ext4: reduce one ...
218
  			}
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
219
  		}
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
220
221
222
  	}
  
  	switch (hinfo->hash_version) {
f99b25897   Theodore Ts'o   ext4: Add support...
223
224
225
  	case DX_HASH_LEGACY_UNSIGNED:
  		hash = dx_hack_hash_unsigned(name, len);
  		break;
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
226
  	case DX_HASH_LEGACY:
f99b25897   Theodore Ts'o   ext4: Add support...
227
  		hash = dx_hack_hash_signed(name, len);
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
228
  		break;
f99b25897   Theodore Ts'o   ext4: Add support...
229
230
  	case DX_HASH_HALF_MD4_UNSIGNED:
  		str2hashbuf = str2hashbuf_unsigned;
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
231
232
233
  	case DX_HASH_HALF_MD4:
  		p = name;
  		while (len > 0) {
f99b25897   Theodore Ts'o   ext4: Add support...
234
  			(*str2hashbuf)(p, len, in, 8);
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
235
236
237
238
239
240
241
  			half_md4_transform(buf, in);
  			len -= 32;
  			p += 32;
  		}
  		minor_hash = buf[2];
  		hash = buf[1];
  		break;
f99b25897   Theodore Ts'o   ext4: Add support...
242
243
  	case DX_HASH_TEA_UNSIGNED:
  		str2hashbuf = str2hashbuf_unsigned;
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
244
245
246
  	case DX_HASH_TEA:
  		p = name;
  		while (len > 0) {
f99b25897   Theodore Ts'o   ext4: Add support...
247
  			(*str2hashbuf)(p, len, in, 4);
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
248
249
250
251
252
253
254
255
256
257
258
259
  			TEA_transform(buf, in);
  			len -= 16;
  			p += 16;
  		}
  		hash = buf[0];
  		minor_hash = buf[1];
  		break;
  	default:
  		hinfo->hash = 0;
  		return -1;
  	}
  	hash = hash & ~1;
d1f5273e9   Fan Yong   ext4: return 32/6...
260
261
  	if (hash == (EXT4_HTREE_EOF_32BIT << 1))
  		hash = (EXT4_HTREE_EOF_32BIT - 1) << 1;
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
262
263
264
265
  	hinfo->hash = hash;
  	hinfo->minor_hash = minor_hash;
  	return 0;
  }