Merge tag 'gfs2-merge-window' of git://git.kernel.org/pub/scm/linux/kernel/git/gfs2...
[linux-2.6-microblaze.git] / crypto / cast5_generic.c
1 /* Kernel cryptographic api.
2 * cast5.c - Cast5 cipher algorithm (rfc2144).
3 *
4 * Derived from GnuPG implementation of cast5.
5 *
6 * Major Changes.
7 *       Complete conformance to rfc2144.
8 *       Supports key size from 40 to 128 bits.
9 *
10 * Copyright (C) 1998, 1999, 2000, 2001 Free Software Foundation, Inc.
11 * Copyright (C) 2003 Kartikey Mahendra Bhatt <kartik_me@hotmail.com>.
12 *
13 * This program is free software; you can redistribute it and/or modify it
14 * under the terms of GNU General Public License as published by the Free
15 * Software Foundation; either version 2 of the License, or (at your option)
16 * any later version.
17 *
18 * You should have received a copy of the GNU General Public License
19 * along with this program; if not, write to the Free Software
20 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
21 */
22
23
24 #include <asm/byteorder.h>
25 #include <linux/init.h>
26 #include <linux/crypto.h>
27 #include <linux/module.h>
28 #include <linux/errno.h>
29 #include <linux/string.h>
30 #include <linux/types.h>
31 #include <crypto/cast5.h>
32
33 static const u32 s5[256] = {
34         0x7ec90c04, 0x2c6e74b9, 0x9b0e66df, 0xa6337911, 0xb86a7fff,
35         0x1dd358f5, 0x44dd9d44, 0x1731167f,
36         0x08fbf1fa, 0xe7f511cc, 0xd2051b00, 0x735aba00, 0x2ab722d8,
37         0x386381cb, 0xacf6243a, 0x69befd7a,
38         0xe6a2e77f, 0xf0c720cd, 0xc4494816, 0xccf5c180, 0x38851640,
39         0x15b0a848, 0xe68b18cb, 0x4caadeff,
40         0x5f480a01, 0x0412b2aa, 0x259814fc, 0x41d0efe2, 0x4e40b48d,
41         0x248eb6fb, 0x8dba1cfe, 0x41a99b02,
42         0x1a550a04, 0xba8f65cb, 0x7251f4e7, 0x95a51725, 0xc106ecd7,
43         0x97a5980a, 0xc539b9aa, 0x4d79fe6a,
44         0xf2f3f763, 0x68af8040, 0xed0c9e56, 0x11b4958b, 0xe1eb5a88,
45         0x8709e6b0, 0xd7e07156, 0x4e29fea7,
46         0x6366e52d, 0x02d1c000, 0xc4ac8e05, 0x9377f571, 0x0c05372a,
47         0x578535f2, 0x2261be02, 0xd642a0c9,
48         0xdf13a280, 0x74b55bd2, 0x682199c0, 0xd421e5ec, 0x53fb3ce8,
49         0xc8adedb3, 0x28a87fc9, 0x3d959981,
50         0x5c1ff900, 0xfe38d399, 0x0c4eff0b, 0x062407ea, 0xaa2f4fb1,
51         0x4fb96976, 0x90c79505, 0xb0a8a774,
52         0xef55a1ff, 0xe59ca2c2, 0xa6b62d27, 0xe66a4263, 0xdf65001f,
53         0x0ec50966, 0xdfdd55bc, 0x29de0655,
54         0x911e739a, 0x17af8975, 0x32c7911c, 0x89f89468, 0x0d01e980,
55         0x524755f4, 0x03b63cc9, 0x0cc844b2,
56         0xbcf3f0aa, 0x87ac36e9, 0xe53a7426, 0x01b3d82b, 0x1a9e7449,
57         0x64ee2d7e, 0xcddbb1da, 0x01c94910,
58         0xb868bf80, 0x0d26f3fd, 0x9342ede7, 0x04a5c284, 0x636737b6,
59         0x50f5b616, 0xf24766e3, 0x8eca36c1,
60         0x136e05db, 0xfef18391, 0xfb887a37, 0xd6e7f7d4, 0xc7fb7dc9,
61         0x3063fcdf, 0xb6f589de, 0xec2941da,
62         0x26e46695, 0xb7566419, 0xf654efc5, 0xd08d58b7, 0x48925401,
63         0xc1bacb7f, 0xe5ff550f, 0xb6083049,
64         0x5bb5d0e8, 0x87d72e5a, 0xab6a6ee1, 0x223a66ce, 0xc62bf3cd,
65         0x9e0885f9, 0x68cb3e47, 0x086c010f,
66         0xa21de820, 0xd18b69de, 0xf3f65777, 0xfa02c3f6, 0x407edac3,
67         0xcbb3d550, 0x1793084d, 0xb0d70eba,
68         0x0ab378d5, 0xd951fb0c, 0xded7da56, 0x4124bbe4, 0x94ca0b56,
69         0x0f5755d1, 0xe0e1e56e, 0x6184b5be,
70         0x580a249f, 0x94f74bc0, 0xe327888e, 0x9f7b5561, 0xc3dc0280,
71         0x05687715, 0x646c6bd7, 0x44904db3,
72         0x66b4f0a3, 0xc0f1648a, 0x697ed5af, 0x49e92ff6, 0x309e374f,
73         0x2cb6356a, 0x85808573, 0x4991f840,
74         0x76f0ae02, 0x083be84d, 0x28421c9a, 0x44489406, 0x736e4cb8,
75         0xc1092910, 0x8bc95fc6, 0x7d869cf4,
76         0x134f616f, 0x2e77118d, 0xb31b2be1, 0xaa90b472, 0x3ca5d717,
77         0x7d161bba, 0x9cad9010, 0xaf462ba2,
78         0x9fe459d2, 0x45d34559, 0xd9f2da13, 0xdbc65487, 0xf3e4f94e,
79         0x176d486f, 0x097c13ea, 0x631da5c7,
80         0x445f7382, 0x175683f4, 0xcdc66a97, 0x70be0288, 0xb3cdcf72,
81         0x6e5dd2f3, 0x20936079, 0x459b80a5,
82         0xbe60e2db, 0xa9c23101, 0xeba5315c, 0x224e42f2, 0x1c5c1572,
83         0xf6721b2c, 0x1ad2fff3, 0x8c25404e,
84         0x324ed72f, 0x4067b7fd, 0x0523138e, 0x5ca3bc78, 0xdc0fd66e,
85         0x75922283, 0x784d6b17, 0x58ebb16e,
86         0x44094f85, 0x3f481d87, 0xfcfeae7b, 0x77b5ff76, 0x8c2302bf,
87         0xaaf47556, 0x5f46b02a, 0x2b092801,
88         0x3d38f5f7, 0x0ca81f36, 0x52af4a8a, 0x66d5e7c0, 0xdf3b0874,
89         0x95055110, 0x1b5ad7a8, 0xf61ed5ad,
90         0x6cf6e479, 0x20758184, 0xd0cefa65, 0x88f7be58, 0x4a046826,
91         0x0ff6f8f3, 0xa09c7f70, 0x5346aba0,
92         0x5ce96c28, 0xe176eda3, 0x6bac307f, 0x376829d2, 0x85360fa9,
93         0x17e3fe2a, 0x24b79767, 0xf5a96b20,
94         0xd6cd2595, 0x68ff1ebf, 0x7555442c, 0xf19f06be, 0xf9e0659a,
95         0xeeb9491d, 0x34010718, 0xbb30cab8,
96         0xe822fe15, 0x88570983, 0x750e6249, 0xda627e55, 0x5e76ffa8,
97         0xb1534546, 0x6d47de08, 0xefe9e7d4
98 };
99 static const u32 s6[256] = {
100         0xf6fa8f9d, 0x2cac6ce1, 0x4ca34867, 0xe2337f7c, 0x95db08e7,
101         0x016843b4, 0xeced5cbc, 0x325553ac,
102         0xbf9f0960, 0xdfa1e2ed, 0x83f0579d, 0x63ed86b9, 0x1ab6a6b8,
103         0xde5ebe39, 0xf38ff732, 0x8989b138,
104         0x33f14961, 0xc01937bd, 0xf506c6da, 0xe4625e7e, 0xa308ea99,
105         0x4e23e33c, 0x79cbd7cc, 0x48a14367,
106         0xa3149619, 0xfec94bd5, 0xa114174a, 0xeaa01866, 0xa084db2d,
107         0x09a8486f, 0xa888614a, 0x2900af98,
108         0x01665991, 0xe1992863, 0xc8f30c60, 0x2e78ef3c, 0xd0d51932,
109         0xcf0fec14, 0xf7ca07d2, 0xd0a82072,
110         0xfd41197e, 0x9305a6b0, 0xe86be3da, 0x74bed3cd, 0x372da53c,
111         0x4c7f4448, 0xdab5d440, 0x6dba0ec3,
112         0x083919a7, 0x9fbaeed9, 0x49dbcfb0, 0x4e670c53, 0x5c3d9c01,
113         0x64bdb941, 0x2c0e636a, 0xba7dd9cd,
114         0xea6f7388, 0xe70bc762, 0x35f29adb, 0x5c4cdd8d, 0xf0d48d8c,
115         0xb88153e2, 0x08a19866, 0x1ae2eac8,
116         0x284caf89, 0xaa928223, 0x9334be53, 0x3b3a21bf, 0x16434be3,
117         0x9aea3906, 0xefe8c36e, 0xf890cdd9,
118         0x80226dae, 0xc340a4a3, 0xdf7e9c09, 0xa694a807, 0x5b7c5ecc,
119         0x221db3a6, 0x9a69a02f, 0x68818a54,
120         0xceb2296f, 0x53c0843a, 0xfe893655, 0x25bfe68a, 0xb4628abc,
121         0xcf222ebf, 0x25ac6f48, 0xa9a99387,
122         0x53bddb65, 0xe76ffbe7, 0xe967fd78, 0x0ba93563, 0x8e342bc1,
123         0xe8a11be9, 0x4980740d, 0xc8087dfc,
124         0x8de4bf99, 0xa11101a0, 0x7fd37975, 0xda5a26c0, 0xe81f994f,
125         0x9528cd89, 0xfd339fed, 0xb87834bf,
126         0x5f04456d, 0x22258698, 0xc9c4c83b, 0x2dc156be, 0x4f628daa,
127         0x57f55ec5, 0xe2220abe, 0xd2916ebf,
128         0x4ec75b95, 0x24f2c3c0, 0x42d15d99, 0xcd0d7fa0, 0x7b6e27ff,
129         0xa8dc8af0, 0x7345c106, 0xf41e232f,
130         0x35162386, 0xe6ea8926, 0x3333b094, 0x157ec6f2, 0x372b74af,
131         0x692573e4, 0xe9a9d848, 0xf3160289,
132         0x3a62ef1d, 0xa787e238, 0xf3a5f676, 0x74364853, 0x20951063,
133         0x4576698d, 0xb6fad407, 0x592af950,
134         0x36f73523, 0x4cfb6e87, 0x7da4cec0, 0x6c152daa, 0xcb0396a8,
135         0xc50dfe5d, 0xfcd707ab, 0x0921c42f,
136         0x89dff0bb, 0x5fe2be78, 0x448f4f33, 0x754613c9, 0x2b05d08d,
137         0x48b9d585, 0xdc049441, 0xc8098f9b,
138         0x7dede786, 0xc39a3373, 0x42410005, 0x6a091751, 0x0ef3c8a6,
139         0x890072d6, 0x28207682, 0xa9a9f7be,
140         0xbf32679d, 0xd45b5b75, 0xb353fd00, 0xcbb0e358, 0x830f220a,
141         0x1f8fb214, 0xd372cf08, 0xcc3c4a13,
142         0x8cf63166, 0x061c87be, 0x88c98f88, 0x6062e397, 0x47cf8e7a,
143         0xb6c85283, 0x3cc2acfb, 0x3fc06976,
144         0x4e8f0252, 0x64d8314d, 0xda3870e3, 0x1e665459, 0xc10908f0,
145         0x513021a5, 0x6c5b68b7, 0x822f8aa0,
146         0x3007cd3e, 0x74719eef, 0xdc872681, 0x073340d4, 0x7e432fd9,
147         0x0c5ec241, 0x8809286c, 0xf592d891,
148         0x08a930f6, 0x957ef305, 0xb7fbffbd, 0xc266e96f, 0x6fe4ac98,
149         0xb173ecc0, 0xbc60b42a, 0x953498da,
150         0xfba1ae12, 0x2d4bd736, 0x0f25faab, 0xa4f3fceb, 0xe2969123,
151         0x257f0c3d, 0x9348af49, 0x361400bc,
152         0xe8816f4a, 0x3814f200, 0xa3f94043, 0x9c7a54c2, 0xbc704f57,
153         0xda41e7f9, 0xc25ad33a, 0x54f4a084,
154         0xb17f5505, 0x59357cbe, 0xedbd15c8, 0x7f97c5ab, 0xba5ac7b5,
155         0xb6f6deaf, 0x3a479c3a, 0x5302da25,
156         0x653d7e6a, 0x54268d49, 0x51a477ea, 0x5017d55b, 0xd7d25d88,
157         0x44136c76, 0x0404a8c8, 0xb8e5a121,
158         0xb81a928a, 0x60ed5869, 0x97c55b96, 0xeaec991b, 0x29935913,
159         0x01fdb7f1, 0x088e8dfa, 0x9ab6f6f5,
160         0x3b4cbf9f, 0x4a5de3ab, 0xe6051d35, 0xa0e1d855, 0xd36b4cf1,
161         0xf544edeb, 0xb0e93524, 0xbebb8fbd,
162         0xa2d762cf, 0x49c92f54, 0x38b5f331, 0x7128a454, 0x48392905,
163         0xa65b1db8, 0x851c97bd, 0xd675cf2f
164 };
165 static const u32 s7[256] = {
166         0x85e04019, 0x332bf567, 0x662dbfff, 0xcfc65693, 0x2a8d7f6f,
167         0xab9bc912, 0xde6008a1, 0x2028da1f,
168         0x0227bce7, 0x4d642916, 0x18fac300, 0x50f18b82, 0x2cb2cb11,
169         0xb232e75c, 0x4b3695f2, 0xb28707de,
170         0xa05fbcf6, 0xcd4181e9, 0xe150210c, 0xe24ef1bd, 0xb168c381,
171         0xfde4e789, 0x5c79b0d8, 0x1e8bfd43,
172         0x4d495001, 0x38be4341, 0x913cee1d, 0x92a79c3f, 0x089766be,
173         0xbaeeadf4, 0x1286becf, 0xb6eacb19,
174         0x2660c200, 0x7565bde4, 0x64241f7a, 0x8248dca9, 0xc3b3ad66,
175         0x28136086, 0x0bd8dfa8, 0x356d1cf2,
176         0x107789be, 0xb3b2e9ce, 0x0502aa8f, 0x0bc0351e, 0x166bf52a,
177         0xeb12ff82, 0xe3486911, 0xd34d7516,
178         0x4e7b3aff, 0x5f43671b, 0x9cf6e037, 0x4981ac83, 0x334266ce,
179         0x8c9341b7, 0xd0d854c0, 0xcb3a6c88,
180         0x47bc2829, 0x4725ba37, 0xa66ad22b, 0x7ad61f1e, 0x0c5cbafa,
181         0x4437f107, 0xb6e79962, 0x42d2d816,
182         0x0a961288, 0xe1a5c06e, 0x13749e67, 0x72fc081a, 0xb1d139f7,
183         0xf9583745, 0xcf19df58, 0xbec3f756,
184         0xc06eba30, 0x07211b24, 0x45c28829, 0xc95e317f, 0xbc8ec511,
185         0x38bc46e9, 0xc6e6fa14, 0xbae8584a,
186         0xad4ebc46, 0x468f508b, 0x7829435f, 0xf124183b, 0x821dba9f,
187         0xaff60ff4, 0xea2c4e6d, 0x16e39264,
188         0x92544a8b, 0x009b4fc3, 0xaba68ced, 0x9ac96f78, 0x06a5b79a,
189         0xb2856e6e, 0x1aec3ca9, 0xbe838688,
190         0x0e0804e9, 0x55f1be56, 0xe7e5363b, 0xb3a1f25d, 0xf7debb85,
191         0x61fe033c, 0x16746233, 0x3c034c28,
192         0xda6d0c74, 0x79aac56c, 0x3ce4e1ad, 0x51f0c802, 0x98f8f35a,
193         0x1626a49f, 0xeed82b29, 0x1d382fe3,
194         0x0c4fb99a, 0xbb325778, 0x3ec6d97b, 0x6e77a6a9, 0xcb658b5c,
195         0xd45230c7, 0x2bd1408b, 0x60c03eb7,
196         0xb9068d78, 0xa33754f4, 0xf430c87d, 0xc8a71302, 0xb96d8c32,
197         0xebd4e7be, 0xbe8b9d2d, 0x7979fb06,
198         0xe7225308, 0x8b75cf77, 0x11ef8da4, 0xe083c858, 0x8d6b786f,
199         0x5a6317a6, 0xfa5cf7a0, 0x5dda0033,
200         0xf28ebfb0, 0xf5b9c310, 0xa0eac280, 0x08b9767a, 0xa3d9d2b0,
201         0x79d34217, 0x021a718d, 0x9ac6336a,
202         0x2711fd60, 0x438050e3, 0x069908a8, 0x3d7fedc4, 0x826d2bef,
203         0x4eeb8476, 0x488dcf25, 0x36c9d566,
204         0x28e74e41, 0xc2610aca, 0x3d49a9cf, 0xbae3b9df, 0xb65f8de6,
205         0x92aeaf64, 0x3ac7d5e6, 0x9ea80509,
206         0xf22b017d, 0xa4173f70, 0xdd1e16c3, 0x15e0d7f9, 0x50b1b887,
207         0x2b9f4fd5, 0x625aba82, 0x6a017962,
208         0x2ec01b9c, 0x15488aa9, 0xd716e740, 0x40055a2c, 0x93d29a22,
209         0xe32dbf9a, 0x058745b9, 0x3453dc1e,
210         0xd699296e, 0x496cff6f, 0x1c9f4986, 0xdfe2ed07, 0xb87242d1,
211         0x19de7eae, 0x053e561a, 0x15ad6f8c,
212         0x66626c1c, 0x7154c24c, 0xea082b2a, 0x93eb2939, 0x17dcb0f0,
213         0x58d4f2ae, 0x9ea294fb, 0x52cf564c,
214         0x9883fe66, 0x2ec40581, 0x763953c3, 0x01d6692e, 0xd3a0c108,
215         0xa1e7160e, 0xe4f2dfa6, 0x693ed285,
216         0x74904698, 0x4c2b0edd, 0x4f757656, 0x5d393378, 0xa132234f,
217         0x3d321c5d, 0xc3f5e194, 0x4b269301,
218         0xc79f022f, 0x3c997e7e, 0x5e4f9504, 0x3ffafbbd, 0x76f7ad0e,
219         0x296693f4, 0x3d1fce6f, 0xc61e45be,
220         0xd3b5ab34, 0xf72bf9b7, 0x1b0434c0, 0x4e72b567, 0x5592a33d,
221         0xb5229301, 0xcfd2a87f, 0x60aeb767,
222         0x1814386b, 0x30bcc33d, 0x38a0c07d, 0xfd1606f2, 0xc363519b,
223         0x589dd390, 0x5479f8e6, 0x1cb8d647,
224         0x97fd61a9, 0xea7759f4, 0x2d57539d, 0x569a58cf, 0xe84e63ad,
225         0x462e1b78, 0x6580f87e, 0xf3817914,
226         0x91da55f4, 0x40a230f3, 0xd1988f35, 0xb6e318d2, 0x3ffa50bc,
227         0x3d40f021, 0xc3c0bdae, 0x4958c24c,
228         0x518f36b2, 0x84b1d370, 0x0fedce83, 0x878ddada, 0xf2a279c7,
229         0x94e01be8, 0x90716f4b, 0x954b8aa3
230 };
231 static const u32 sb8[256] = {
232         0xe216300d, 0xbbddfffc, 0xa7ebdabd, 0x35648095, 0x7789f8b7,
233         0xe6c1121b, 0x0e241600, 0x052ce8b5,
234         0x11a9cfb0, 0xe5952f11, 0xece7990a, 0x9386d174, 0x2a42931c,
235         0x76e38111, 0xb12def3a, 0x37ddddfc,
236         0xde9adeb1, 0x0a0cc32c, 0xbe197029, 0x84a00940, 0xbb243a0f,
237         0xb4d137cf, 0xb44e79f0, 0x049eedfd,
238         0x0b15a15d, 0x480d3168, 0x8bbbde5a, 0x669ded42, 0xc7ece831,
239         0x3f8f95e7, 0x72df191b, 0x7580330d,
240         0x94074251, 0x5c7dcdfa, 0xabbe6d63, 0xaa402164, 0xb301d40a,
241         0x02e7d1ca, 0x53571dae, 0x7a3182a2,
242         0x12a8ddec, 0xfdaa335d, 0x176f43e8, 0x71fb46d4, 0x38129022,
243         0xce949ad4, 0xb84769ad, 0x965bd862,
244         0x82f3d055, 0x66fb9767, 0x15b80b4e, 0x1d5b47a0, 0x4cfde06f,
245         0xc28ec4b8, 0x57e8726e, 0x647a78fc,
246         0x99865d44, 0x608bd593, 0x6c200e03, 0x39dc5ff6, 0x5d0b00a3,
247         0xae63aff2, 0x7e8bd632, 0x70108c0c,
248         0xbbd35049, 0x2998df04, 0x980cf42a, 0x9b6df491, 0x9e7edd53,
249         0x06918548, 0x58cb7e07, 0x3b74ef2e,
250         0x522fffb1, 0xd24708cc, 0x1c7e27cd, 0xa4eb215b, 0x3cf1d2e2,
251         0x19b47a38, 0x424f7618, 0x35856039,
252         0x9d17dee7, 0x27eb35e6, 0xc9aff67b, 0x36baf5b8, 0x09c467cd,
253         0xc18910b1, 0xe11dbf7b, 0x06cd1af8,
254         0x7170c608, 0x2d5e3354, 0xd4de495a, 0x64c6d006, 0xbcc0c62c,
255         0x3dd00db3, 0x708f8f34, 0x77d51b42,
256         0x264f620f, 0x24b8d2bf, 0x15c1b79e, 0x46a52564, 0xf8d7e54e,
257         0x3e378160, 0x7895cda5, 0x859c15a5,
258         0xe6459788, 0xc37bc75f, 0xdb07ba0c, 0x0676a3ab, 0x7f229b1e,
259         0x31842e7b, 0x24259fd7, 0xf8bef472,
260         0x835ffcb8, 0x6df4c1f2, 0x96f5b195, 0xfd0af0fc, 0xb0fe134c,
261         0xe2506d3d, 0x4f9b12ea, 0xf215f225,
262         0xa223736f, 0x9fb4c428, 0x25d04979, 0x34c713f8, 0xc4618187,
263         0xea7a6e98, 0x7cd16efc, 0x1436876c,
264         0xf1544107, 0xbedeee14, 0x56e9af27, 0xa04aa441, 0x3cf7c899,
265         0x92ecbae6, 0xdd67016d, 0x151682eb,
266         0xa842eedf, 0xfdba60b4, 0xf1907b75, 0x20e3030f, 0x24d8c29e,
267         0xe139673b, 0xefa63fb8, 0x71873054,
268         0xb6f2cf3b, 0x9f326442, 0xcb15a4cc, 0xb01a4504, 0xf1e47d8d,
269         0x844a1be5, 0xbae7dfdc, 0x42cbda70,
270         0xcd7dae0a, 0x57e85b7a, 0xd53f5af6, 0x20cf4d8c, 0xcea4d428,
271         0x79d130a4, 0x3486ebfb, 0x33d3cddc,
272         0x77853b53, 0x37effcb5, 0xc5068778, 0xe580b3e6, 0x4e68b8f4,
273         0xc5c8b37e, 0x0d809ea2, 0x398feb7c,
274         0x132a4f94, 0x43b7950e, 0x2fee7d1c, 0x223613bd, 0xdd06caa2,
275         0x37df932b, 0xc4248289, 0xacf3ebc3,
276         0x5715f6b7, 0xef3478dd, 0xf267616f, 0xc148cbe4, 0x9052815e,
277         0x5e410fab, 0xb48a2465, 0x2eda7fa4,
278         0xe87b40e4, 0xe98ea084, 0x5889e9e1, 0xefd390fc, 0xdd07d35b,
279         0xdb485694, 0x38d7e5b2, 0x57720101,
280         0x730edebc, 0x5b643113, 0x94917e4f, 0x503c2fba, 0x646f1282,
281         0x7523d24a, 0xe0779695, 0xf9c17a8f,
282         0x7a5b2121, 0xd187b896, 0x29263a4d, 0xba510cdf, 0x81f47c9f,
283         0xad1163ed, 0xea7b5965, 0x1a00726e,
284         0x11403092, 0x00da6d77, 0x4a0cdd61, 0xad1f4603, 0x605bdfb0,
285         0x9eedc364, 0x22ebe6a8, 0xcee7d28a,
286         0xa0e736a0, 0x5564a6b9, 0x10853209, 0xc7eb8f37, 0x2de705ca,
287         0x8951570f, 0xdf09822b, 0xbd691a6c,
288         0xaa12e4f2, 0x87451c0f, 0xe0f6a27a, 0x3ada4819, 0x4cf1764f,
289         0x0d771c2b, 0x67cdb156, 0x350d8384,
290         0x5938fa0f, 0x42399ef3, 0x36997b07, 0x0e84093d, 0x4aa93e61,
291         0x8360d87b, 0x1fa98b0c, 0x1149382c,
292         0xe97625a5, 0x0614d1b7, 0x0e25244b, 0x0c768347, 0x589e8d82,
293         0x0d2059d1, 0xa466bb1e, 0xf8da0a82,
294         0x04f19130, 0xba6e4ec0, 0x99265164, 0x1ee7230d, 0x50b2ad80,
295         0xeaee6801, 0x8db2a283, 0xea8bf59e
296 };
297
298 #define s1 cast_s1
299 #define s2 cast_s2
300 #define s3 cast_s3
301 #define s4 cast_s4
302
303 #define F1(D, m, r)  ((I = ((m) + (D))), (I = rol32(I, (r))),   \
304         (((s1[I >> 24] ^ s2[(I>>16)&0xff]) - s3[(I>>8)&0xff]) + s4[I&0xff]))
305 #define F2(D, m, r)  ((I = ((m) ^ (D))), (I = rol32(I, (r))),   \
306         (((s1[I >> 24] - s2[(I>>16)&0xff]) + s3[(I>>8)&0xff]) ^ s4[I&0xff]))
307 #define F3(D, m, r)  ((I = ((m) - (D))), (I = rol32(I, (r))),   \
308         (((s1[I >> 24] + s2[(I>>16)&0xff]) ^ s3[(I>>8)&0xff]) - s4[I&0xff]))
309
310
311 void __cast5_encrypt(struct cast5_ctx *c, u8 *outbuf, const u8 *inbuf)
312 {
313         const __be32 *src = (const __be32 *)inbuf;
314         __be32 *dst = (__be32 *)outbuf;
315         u32 l, r, t;
316         u32 I;                  /* used by the Fx macros */
317         u32 *Km;
318         u8 *Kr;
319
320         Km = c->Km;
321         Kr = c->Kr;
322
323         /* (L0,R0) <-- (m1...m64).  (Split the plaintext into left and
324          * right 32-bit halves L0 = m1...m32 and R0 = m33...m64.)
325          */
326         l = be32_to_cpu(src[0]);
327         r = be32_to_cpu(src[1]);
328
329         /* (16 rounds) for i from 1 to 16, compute Li and Ri as follows:
330          *  Li = Ri-1;
331          *  Ri = Li-1 ^ f(Ri-1,Kmi,Kri), where f is defined in Section 2.2
332          * Rounds 1, 4, 7, 10, 13, and 16 use f function Type 1.
333          * Rounds 2, 5, 8, 11, and 14 use f function Type 2.
334          * Rounds 3, 6, 9, 12, and 15 use f function Type 3.
335          */
336
337         t = l; l = r; r = t ^ F1(r, Km[0], Kr[0]);
338         t = l; l = r; r = t ^ F2(r, Km[1], Kr[1]);
339         t = l; l = r; r = t ^ F3(r, Km[2], Kr[2]);
340         t = l; l = r; r = t ^ F1(r, Km[3], Kr[3]);
341         t = l; l = r; r = t ^ F2(r, Km[4], Kr[4]);
342         t = l; l = r; r = t ^ F3(r, Km[5], Kr[5]);
343         t = l; l = r; r = t ^ F1(r, Km[6], Kr[6]);
344         t = l; l = r; r = t ^ F2(r, Km[7], Kr[7]);
345         t = l; l = r; r = t ^ F3(r, Km[8], Kr[8]);
346         t = l; l = r; r = t ^ F1(r, Km[9], Kr[9]);
347         t = l; l = r; r = t ^ F2(r, Km[10], Kr[10]);
348         t = l; l = r; r = t ^ F3(r, Km[11], Kr[11]);
349         if (!(c->rr)) {
350                 t = l; l = r; r = t ^ F1(r, Km[12], Kr[12]);
351                 t = l; l = r; r = t ^ F2(r, Km[13], Kr[13]);
352                 t = l; l = r; r = t ^ F3(r, Km[14], Kr[14]);
353                 t = l; l = r; r = t ^ F1(r, Km[15], Kr[15]);
354         }
355
356         /* c1...c64 <-- (R16,L16).  (Exchange final blocks L16, R16 and
357          *  concatenate to form the ciphertext.) */
358         dst[0] = cpu_to_be32(r);
359         dst[1] = cpu_to_be32(l);
360 }
361 EXPORT_SYMBOL_GPL(__cast5_encrypt);
362
363 static void cast5_encrypt(struct crypto_tfm *tfm, u8 *outbuf, const u8 *inbuf)
364 {
365         __cast5_encrypt(crypto_tfm_ctx(tfm), outbuf, inbuf);
366 }
367
368 void __cast5_decrypt(struct cast5_ctx *c, u8 *outbuf, const u8 *inbuf)
369 {
370         const __be32 *src = (const __be32 *)inbuf;
371         __be32 *dst = (__be32 *)outbuf;
372         u32 l, r, t;
373         u32 I;
374         u32 *Km;
375         u8 *Kr;
376
377         Km = c->Km;
378         Kr = c->Kr;
379
380         l = be32_to_cpu(src[0]);
381         r = be32_to_cpu(src[1]);
382
383         if (!(c->rr)) {
384                 t = l; l = r; r = t ^ F1(r, Km[15], Kr[15]);
385                 t = l; l = r; r = t ^ F3(r, Km[14], Kr[14]);
386                 t = l; l = r; r = t ^ F2(r, Km[13], Kr[13]);
387                 t = l; l = r; r = t ^ F1(r, Km[12], Kr[12]);
388         }
389         t = l; l = r; r = t ^ F3(r, Km[11], Kr[11]);
390         t = l; l = r; r = t ^ F2(r, Km[10], Kr[10]);
391         t = l; l = r; r = t ^ F1(r, Km[9], Kr[9]);
392         t = l; l = r; r = t ^ F3(r, Km[8], Kr[8]);
393         t = l; l = r; r = t ^ F2(r, Km[7], Kr[7]);
394         t = l; l = r; r = t ^ F1(r, Km[6], Kr[6]);
395         t = l; l = r; r = t ^ F3(r, Km[5], Kr[5]);
396         t = l; l = r; r = t ^ F2(r, Km[4], Kr[4]);
397         t = l; l = r; r = t ^ F1(r, Km[3], Kr[3]);
398         t = l; l = r; r = t ^ F3(r, Km[2], Kr[2]);
399         t = l; l = r; r = t ^ F2(r, Km[1], Kr[1]);
400         t = l; l = r; r = t ^ F1(r, Km[0], Kr[0]);
401
402         dst[0] = cpu_to_be32(r);
403         dst[1] = cpu_to_be32(l);
404 }
405 EXPORT_SYMBOL_GPL(__cast5_decrypt);
406
407 static void cast5_decrypt(struct crypto_tfm *tfm, u8 *outbuf, const u8 *inbuf)
408 {
409         __cast5_decrypt(crypto_tfm_ctx(tfm), outbuf, inbuf);
410 }
411
412 static void key_schedule(u32 *x, u32 *z, u32 *k)
413 {
414
415 #define xi(i)   ((x[(i)/4] >> (8*(3-((i)%4)))) & 0xff)
416 #define zi(i)   ((z[(i)/4] >> (8*(3-((i)%4)))) & 0xff)
417
418         z[0] = x[0] ^ s5[xi(13)] ^ s6[xi(15)] ^ s7[xi(12)] ^ sb8[xi(14)] ^
419             s7[xi(8)];
420         z[1] = x[2] ^ s5[zi(0)] ^ s6[zi(2)] ^ s7[zi(1)] ^ sb8[zi(3)] ^
421             sb8[xi(10)];
422         z[2] = x[3] ^ s5[zi(7)] ^ s6[zi(6)] ^ s7[zi(5)] ^ sb8[zi(4)] ^
423             s5[xi(9)];
424         z[3] = x[1] ^ s5[zi(10)] ^ s6[zi(9)] ^ s7[zi(11)] ^ sb8[zi(8)] ^
425             s6[xi(11)];
426         k[0] = s5[zi(8)] ^ s6[zi(9)] ^ s7[zi(7)] ^ sb8[zi(6)] ^ s5[zi(2)];
427         k[1] = s5[zi(10)] ^ s6[zi(11)] ^ s7[zi(5)] ^ sb8[zi(4)] ^
428             s6[zi(6)];
429         k[2] = s5[zi(12)] ^ s6[zi(13)] ^ s7[zi(3)] ^ sb8[zi(2)] ^
430             s7[zi(9)];
431         k[3] = s5[zi(14)] ^ s6[zi(15)] ^ s7[zi(1)] ^ sb8[zi(0)] ^
432             sb8[zi(12)];
433
434         x[0] = z[2] ^ s5[zi(5)] ^ s6[zi(7)] ^ s7[zi(4)] ^ sb8[zi(6)] ^
435             s7[zi(0)];
436         x[1] = z[0] ^ s5[xi(0)] ^ s6[xi(2)] ^ s7[xi(1)] ^ sb8[xi(3)] ^
437             sb8[zi(2)];
438         x[2] = z[1] ^ s5[xi(7)] ^ s6[xi(6)] ^ s7[xi(5)] ^ sb8[xi(4)] ^
439             s5[zi(1)];
440         x[3] = z[3] ^ s5[xi(10)] ^ s6[xi(9)] ^ s7[xi(11)] ^ sb8[xi(8)] ^
441             s6[zi(3)];
442         k[4] = s5[xi(3)] ^ s6[xi(2)] ^ s7[xi(12)] ^ sb8[xi(13)] ^
443             s5[xi(8)];
444         k[5] = s5[xi(1)] ^ s6[xi(0)] ^ s7[xi(14)] ^ sb8[xi(15)] ^
445             s6[xi(13)];
446         k[6] = s5[xi(7)] ^ s6[xi(6)] ^ s7[xi(8)] ^ sb8[xi(9)] ^ s7[xi(3)];
447         k[7] = s5[xi(5)] ^ s6[xi(4)] ^ s7[xi(10)] ^ sb8[xi(11)] ^
448             sb8[xi(7)];
449
450         z[0] = x[0] ^ s5[xi(13)] ^ s6[xi(15)] ^ s7[xi(12)] ^ sb8[xi(14)] ^
451             s7[xi(8)];
452         z[1] = x[2] ^ s5[zi(0)] ^ s6[zi(2)] ^ s7[zi(1)] ^ sb8[zi(3)] ^
453             sb8[xi(10)];
454         z[2] = x[3] ^ s5[zi(7)] ^ s6[zi(6)] ^ s7[zi(5)] ^ sb8[zi(4)] ^
455             s5[xi(9)];
456         z[3] = x[1] ^ s5[zi(10)] ^ s6[zi(9)] ^ s7[zi(11)] ^ sb8[zi(8)] ^
457             s6[xi(11)];
458         k[8] = s5[zi(3)] ^ s6[zi(2)] ^ s7[zi(12)] ^ sb8[zi(13)] ^
459             s5[zi(9)];
460         k[9] = s5[zi(1)] ^ s6[zi(0)] ^ s7[zi(14)] ^ sb8[zi(15)] ^
461             s6[zi(12)];
462         k[10] = s5[zi(7)] ^ s6[zi(6)] ^ s7[zi(8)] ^ sb8[zi(9)] ^ s7[zi(2)];
463         k[11] = s5[zi(5)] ^ s6[zi(4)] ^ s7[zi(10)] ^ sb8[zi(11)] ^
464             sb8[zi(6)];
465
466         x[0] = z[2] ^ s5[zi(5)] ^ s6[zi(7)] ^ s7[zi(4)] ^ sb8[zi(6)] ^
467             s7[zi(0)];
468         x[1] = z[0] ^ s5[xi(0)] ^ s6[xi(2)] ^ s7[xi(1)] ^ sb8[xi(3)] ^
469             sb8[zi(2)];
470         x[2] = z[1] ^ s5[xi(7)] ^ s6[xi(6)] ^ s7[xi(5)] ^ sb8[xi(4)] ^
471             s5[zi(1)];
472         x[3] = z[3] ^ s5[xi(10)] ^ s6[xi(9)] ^ s7[xi(11)] ^ sb8[xi(8)] ^
473             s6[zi(3)];
474         k[12] = s5[xi(8)] ^ s6[xi(9)] ^ s7[xi(7)] ^ sb8[xi(6)] ^ s5[xi(3)];
475         k[13] = s5[xi(10)] ^ s6[xi(11)] ^ s7[xi(5)] ^ sb8[xi(4)] ^
476             s6[xi(7)];
477         k[14] = s5[xi(12)] ^ s6[xi(13)] ^ s7[xi(3)] ^ sb8[xi(2)] ^
478             s7[xi(8)];
479         k[15] = s5[xi(14)] ^ s6[xi(15)] ^ s7[xi(1)] ^ sb8[xi(0)] ^
480             sb8[xi(13)];
481
482 #undef xi
483 #undef zi
484 }
485
486
487 int cast5_setkey(struct crypto_tfm *tfm, const u8 *key, unsigned int key_len)
488 {
489         struct cast5_ctx *c = crypto_tfm_ctx(tfm);
490         int i;
491         u32 x[4];
492         u32 z[4];
493         u32 k[16];
494         __be32 p_key[4];
495
496         c->rr = key_len <= 10 ? 1 : 0;
497
498         memset(p_key, 0, 16);
499         memcpy(p_key, key, key_len);
500
501
502         x[0] = be32_to_cpu(p_key[0]);
503         x[1] = be32_to_cpu(p_key[1]);
504         x[2] = be32_to_cpu(p_key[2]);
505         x[3] = be32_to_cpu(p_key[3]);
506
507         key_schedule(x, z, k);
508         for (i = 0; i < 16; i++)
509                 c->Km[i] = k[i];
510         key_schedule(x, z, k);
511         for (i = 0; i < 16; i++)
512                 c->Kr[i] = k[i] & 0x1f;
513         return 0;
514 }
515 EXPORT_SYMBOL_GPL(cast5_setkey);
516
517 static struct crypto_alg alg = {
518         .cra_name               = "cast5",
519         .cra_driver_name        = "cast5-generic",
520         .cra_priority           = 100,
521         .cra_flags              = CRYPTO_ALG_TYPE_CIPHER,
522         .cra_blocksize          = CAST5_BLOCK_SIZE,
523         .cra_ctxsize            = sizeof(struct cast5_ctx),
524         .cra_alignmask          = 3,
525         .cra_module             = THIS_MODULE,
526         .cra_u                  = {
527                 .cipher = {
528                         .cia_min_keysize = CAST5_MIN_KEY_SIZE,
529                         .cia_max_keysize = CAST5_MAX_KEY_SIZE,
530                         .cia_setkey  = cast5_setkey,
531                         .cia_encrypt = cast5_encrypt,
532                         .cia_decrypt = cast5_decrypt
533                 }
534         }
535 };
536
537 static int __init cast5_mod_init(void)
538 {
539         return crypto_register_alg(&alg);
540 }
541
542 static void __exit cast5_mod_fini(void)
543 {
544         crypto_unregister_alg(&alg);
545 }
546
547 module_init(cast5_mod_init);
548 module_exit(cast5_mod_fini);
549
550 MODULE_LICENSE("GPL");
551 MODULE_DESCRIPTION("Cast5 Cipher Algorithm");
552 MODULE_ALIAS_CRYPTO("cast5");
553 MODULE_ALIAS_CRYPTO("cast5-generic");