Blame view

fs/ext4/hash.c 4.33 KB
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
1
  /*
617ba13b3   Mingming Cao   [PATCH] ext4: ren...
2
   *  linux/fs/ext4/hash.c
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
3
4
5
6
7
8
9
10
11
12
   *
   * Copyright (C) 2002 by Theodore Ts'o
   *
   * This file is released under the GPL v2.
   *
   * This file may be redistributed under the terms of the GNU Public
   * License.
   */
  
  #include <linux/fs.h>
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
13
  #include <linux/cryptohash.h>
3dcf54515   Christoph Hellwig   ext4: move header...
14
  #include "ext4.h"
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
15
16
17
18
19
20
21
22
23
24
25
26
27
28
  
  #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...
29
  	} while (--n);
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
30
31
32
33
34
35
36
  
  	buf[0] += b0;
  	buf[1] += b1;
  }
  
  
  /* The old legacy hash */
f99b25897   Theodore Ts'o   ext4: Add support...
37
  static __u32 dx_hack_hash_unsigned(const char *name, int len)
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
38
  {
f99b25897   Theodore Ts'o   ext4: Add support...
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
  	__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...
57
  	while (len--) {
f99b25897   Theodore Ts'o   ext4: Add support...
58
  		hash = hash1 + (hash0 ^ (((int) *scp++) * 7152373));
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
59

f99b25897   Theodore Ts'o   ext4: Add support...
60
61
  		if (hash & 0x80000000)
  			hash -= 0x7fffffff;
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
62
63
64
  		hash1 = hash0;
  		hash0 = hash;
  	}
f99b25897   Theodore Ts'o   ext4: Add support...
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
  	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++) {
  		if ((i % 4) == 0)
  			val = pad;
  		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...
94
  }
f99b25897   Theodore Ts'o   ext4: Add support...
95
  static void str2hashbuf_unsigned(const char *msg, int len, __u32 *buf, int num)
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
96
97
98
  {
  	__u32	pad, val;
  	int	i;
f99b25897   Theodore Ts'o   ext4: Add support...
99
  	const unsigned char *ucp = (const unsigned char *) msg;
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
100
101
102
103
104
105
106
  
  	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...
107
  	for (i = 0; i < len; i++) {
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
108
109
  		if ((i % 4) == 0)
  			val = pad;
f99b25897   Theodore Ts'o   ext4: Add support...
110
  		val = ((int) ucp[i]) + (val << 8);
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
  		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...
136
  int ext4fs_dirhash(const char *name, int len, struct dx_hash_info *hinfo)
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
137
138
139
140
141
142
  {
  	__u32	hash;
  	__u32	minor_hash = 0;
  	const char	*p;
  	int		i;
  	__u32		in[8], buf[4];
f99b25897   Theodore Ts'o   ext4: Add support...
143
144
  	void		(*str2hashbuf)(const char *, int, __u32 *, int) =
  				str2hashbuf_signed;
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
145
146
147
148
149
150
151
152
153
  
  	/* 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...
154
  		for (i = 0; i < 4; i++) {
0e79537d3   Cong Ding   ext4: reduce one ...
155
156
  			if (hinfo->seed[i]) {
  				memcpy(buf, hinfo->seed, sizeof(buf));
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
157
  				break;
0e79537d3   Cong Ding   ext4: reduce one ...
158
  			}
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
159
  		}
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
160
161
162
  	}
  
  	switch (hinfo->hash_version) {
f99b25897   Theodore Ts'o   ext4: Add support...
163
164
165
  	case DX_HASH_LEGACY_UNSIGNED:
  		hash = dx_hack_hash_unsigned(name, len);
  		break;
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
166
  	case DX_HASH_LEGACY:
f99b25897   Theodore Ts'o   ext4: Add support...
167
  		hash = dx_hack_hash_signed(name, len);
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
168
  		break;
f99b25897   Theodore Ts'o   ext4: Add support...
169
170
  	case DX_HASH_HALF_MD4_UNSIGNED:
  		str2hashbuf = str2hashbuf_unsigned;
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
171
172
173
  	case DX_HASH_HALF_MD4:
  		p = name;
  		while (len > 0) {
f99b25897   Theodore Ts'o   ext4: Add support...
174
  			(*str2hashbuf)(p, len, in, 8);
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
175
176
177
178
179
180
181
  			half_md4_transform(buf, in);
  			len -= 32;
  			p += 32;
  		}
  		minor_hash = buf[2];
  		hash = buf[1];
  		break;
f99b25897   Theodore Ts'o   ext4: Add support...
182
183
  	case DX_HASH_TEA_UNSIGNED:
  		str2hashbuf = str2hashbuf_unsigned;
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
184
185
186
  	case DX_HASH_TEA:
  		p = name;
  		while (len > 0) {
f99b25897   Theodore Ts'o   ext4: Add support...
187
  			(*str2hashbuf)(p, len, in, 4);
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
188
189
190
191
192
193
194
195
196
197
198
199
  			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...
200
201
  	if (hash == (EXT4_HTREE_EOF_32BIT << 1))
  		hash = (EXT4_HTREE_EOF_32BIT - 1) << 1;
ac27a0ec1   Dave Kleikamp   [PATCH] ext4: ini...
202
203
204
205
  	hinfo->hash = hash;
  	hinfo->minor_hash = minor_hash;
  	return 0;
  }