Whamcloud - gitweb
LU-13004 lustre: remove support for KVEC bulk descriptors
[fs/lustre-release.git] / lustre / ptlrpc / gss / gss_krb5_mech.c
1 /*
2  * Modifications for Lustre
3  *
4  * Copyright (c) 2007, 2010, Oracle and/or its affiliates. All rights reserved.
5  *
6  * Copyright (c) 2011, 2015, Intel Corporation.
7  *
8  * Author: Eric Mei <ericm@clusterfs.com>
9  */
10
11 /*
12  *  linux/net/sunrpc/gss_krb5_mech.c
13  *  linux/net/sunrpc/gss_krb5_crypto.c
14  *  linux/net/sunrpc/gss_krb5_seal.c
15  *  linux/net/sunrpc/gss_krb5_seqnum.c
16  *  linux/net/sunrpc/gss_krb5_unseal.c
17  *
18  *  Copyright (c) 2001 The Regents of the University of Michigan.
19  *  All rights reserved.
20  *
21  *  Andy Adamson <andros@umich.edu>
22  *  J. Bruce Fields <bfields@umich.edu>
23  *
24  *  Redistribution and use in source and binary forms, with or without
25  *  modification, are permitted provided that the following conditions
26  *  are met:
27  *
28  *  1. Redistributions of source code must retain the above copyright
29  *     notice, this list of conditions and the following disclaimer.
30  *  2. Redistributions in binary form must reproduce the above copyright
31  *     notice, this list of conditions and the following disclaimer in the
32  *     documentation and/or other materials provided with the distribution.
33  *  3. Neither the name of the University nor the names of its
34  *     contributors may be used to endorse or promote products derived
35  *     from this software without specific prior written permission.
36  *
37  *  THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESS OR IMPLIED
38  *  WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
39  *  MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
40  *  DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
41  *  FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
42  *  CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
43  *  SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
44  *  BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
45  *  LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
46  *  NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
47  *  SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
48  *
49  */
50
51 #define DEBUG_SUBSYSTEM S_SEC
52 #include <linux/init.h>
53 #include <linux/module.h>
54 #include <linux/random.h>
55 #include <linux/slab.h>
56 #include <linux/crypto.h>
57 #include <linux/mutex.h>
58
59 #include <obd.h>
60 #include <obd_class.h>
61 #include <obd_support.h>
62 #include <lustre_net.h>
63 #include <lustre_import.h>
64 #include <lustre_sec.h>
65
66 #include "gss_err.h"
67 #include "gss_internal.h"
68 #include "gss_api.h"
69 #include "gss_asn1.h"
70 #include "gss_krb5.h"
71 #include "gss_crypto.h"
72
73 static DEFINE_SPINLOCK(krb5_seq_lock);
74
75 struct krb5_enctype {
76         char           *ke_dispname;
77         char           *ke_enc_name;            /* linux tfm name */
78         char           *ke_hash_name;           /* linux tfm name */
79         int             ke_enc_mode;            /* linux tfm mode */
80         int             ke_hash_size;           /* checksum size */
81         int             ke_conf_size;           /* confounder size */
82         unsigned int    ke_hash_hmac:1;         /* is hmac? */
83 };
84
85 /*
86  * NOTE: for aes128-cts and aes256-cts, MIT implementation use CTS encryption.
87  * but currently we simply CBC with padding, because linux doesn't support CTS
88  * yet. this need to be fixed in the future.
89  */
90 static struct krb5_enctype enctypes[] = {
91         [ENCTYPE_DES_CBC_RAW] = {               /* des-cbc-md5 */
92                 .ke_dispname    = "des-cbc-md5",
93                 .ke_enc_name    = "cbc(des)",
94                 .ke_hash_name   = "md5",
95                 .ke_hash_size   = 16,
96                 .ke_conf_size   = 8,
97         },
98         [ENCTYPE_DES3_CBC_RAW] = {              /* des3-hmac-sha1 */
99                 .ke_dispname    = "des3-hmac-sha1",
100                 .ke_enc_name    = "cbc(des3_ede)",
101                 .ke_hash_name   = "sha1",
102                 .ke_hash_size   = 20,
103                 .ke_conf_size   = 8,
104                 .ke_hash_hmac   = 1,
105         },
106         [ENCTYPE_AES128_CTS_HMAC_SHA1_96] = {   /* aes128-cts */
107                 .ke_dispname    = "aes128-cts-hmac-sha1-96",
108                 .ke_enc_name    = "cbc(aes)",
109                 .ke_hash_name   = "sha1",
110                 .ke_hash_size   = 12,
111                 .ke_conf_size   = 16,
112                 .ke_hash_hmac   = 1,
113         },
114         [ENCTYPE_AES256_CTS_HMAC_SHA1_96] = {   /* aes256-cts */
115                 .ke_dispname    = "aes256-cts-hmac-sha1-96",
116                 .ke_enc_name    = "cbc(aes)",
117                 .ke_hash_name   = "sha1",
118                 .ke_hash_size   = 12,
119                 .ke_conf_size   = 16,
120                 .ke_hash_hmac   = 1,
121         },
122         [ENCTYPE_ARCFOUR_HMAC] = {              /* arcfour-hmac-md5 */
123                 .ke_dispname    = "arcfour-hmac-md5",
124                 .ke_enc_name    = "ecb(arc4)",
125                 .ke_hash_name   = "md5",
126                 .ke_hash_size   = 16,
127                 .ke_conf_size   = 8,
128                 .ke_hash_hmac   = 1,
129         }
130 };
131
132 static const char * enctype2str(__u32 enctype)
133 {
134         if (enctype < ARRAY_SIZE(enctypes) && enctypes[enctype].ke_dispname)
135                 return enctypes[enctype].ke_dispname;
136
137         return "unknown";
138 }
139
140 static
141 int krb5_init_keys(struct krb5_ctx *kctx)
142 {
143         struct krb5_enctype *ke;
144
145         if (kctx->kc_enctype >= ARRAY_SIZE(enctypes) ||
146             enctypes[kctx->kc_enctype].ke_hash_size == 0) {
147                 CERROR("unsupported enctype %x\n", kctx->kc_enctype);
148                 return -1;
149         }
150
151         ke = &enctypes[kctx->kc_enctype];
152
153         /* tfm arc4 is stateful, user should alloc-use-free by his own */
154         if (kctx->kc_enctype != ENCTYPE_ARCFOUR_HMAC &&
155             gss_keyblock_init(&kctx->kc_keye, ke->ke_enc_name, ke->ke_enc_mode))
156                 return -1;
157
158         /* tfm hmac is stateful, user should alloc-use-free by his own */
159         if (ke->ke_hash_hmac == 0 &&
160             gss_keyblock_init(&kctx->kc_keyi, ke->ke_enc_name, ke->ke_enc_mode))
161                 return -1;
162         if (ke->ke_hash_hmac == 0 &&
163             gss_keyblock_init(&kctx->kc_keyc, ke->ke_enc_name, ke->ke_enc_mode))
164                 return -1;
165
166         return 0;
167 }
168
169 static
170 void delete_context_kerberos(struct krb5_ctx *kctx)
171 {
172         rawobj_free(&kctx->kc_mech_used);
173
174         gss_keyblock_free(&kctx->kc_keye);
175         gss_keyblock_free(&kctx->kc_keyi);
176         gss_keyblock_free(&kctx->kc_keyc);
177 }
178
179 static
180 __u32 import_context_rfc1964(struct krb5_ctx *kctx, char *p, char *end)
181 {
182         unsigned int    tmp_uint, keysize;
183
184         /* seed_init flag */
185         if (gss_get_bytes(&p, end, &tmp_uint, sizeof(tmp_uint)))
186                 goto out_err;
187         kctx->kc_seed_init = (tmp_uint != 0);
188
189         /* seed */
190         if (gss_get_bytes(&p, end, kctx->kc_seed, sizeof(kctx->kc_seed)))
191                 goto out_err;
192
193         /* sign/seal algorithm, not really used now */
194         if (gss_get_bytes(&p, end, &tmp_uint, sizeof(tmp_uint)) ||
195             gss_get_bytes(&p, end, &tmp_uint, sizeof(tmp_uint)))
196                 goto out_err;
197
198         /* end time. While kc_endtime might be 64 bit the krb5 API
199          * still uses 32 bits. To delay the 2038 bug see the incoming
200          * value as a u32 which give us until 2106. See the link for details:
201          *
202          * http://web.mit.edu/kerberos/www/krb5-current/doc/appdev/y2038.html
203          */
204         if (gss_get_bytes(&p, end, &kctx->kc_endtime, sizeof(u32)))
205                 goto out_err;
206
207         /* seq send */
208         if (gss_get_bytes(&p, end, &tmp_uint, sizeof(tmp_uint)))
209                 goto out_err;
210         kctx->kc_seq_send = tmp_uint;
211
212         /* mech oid */
213         if (gss_get_rawobj(&p, end, &kctx->kc_mech_used))
214                 goto out_err;
215
216         /* old style enc/seq keys in format:
217          *   - enctype (u32)
218          *   - keysize (u32)
219          *   - keydata
220          * we decompose them to fit into the new context
221          */
222
223         /* enc key */
224         if (gss_get_bytes(&p, end, &kctx->kc_enctype, sizeof(kctx->kc_enctype)))
225                 goto out_err;
226
227         if (gss_get_bytes(&p, end, &keysize, sizeof(keysize)))
228                 goto out_err;
229
230         if (gss_get_keyblock(&p, end, &kctx->kc_keye, keysize))
231                 goto out_err;
232
233         /* seq key */
234         if (gss_get_bytes(&p, end, &tmp_uint, sizeof(tmp_uint)) ||
235             tmp_uint != kctx->kc_enctype)
236                 goto out_err;
237
238         if (gss_get_bytes(&p, end, &tmp_uint, sizeof(tmp_uint)) ||
239             tmp_uint != keysize)
240                 goto out_err;
241
242         if (gss_get_keyblock(&p, end, &kctx->kc_keyc, keysize))
243                 goto out_err;
244
245         /* old style fallback */
246         if (gss_keyblock_dup(&kctx->kc_keyi, &kctx->kc_keyc))
247                 goto out_err;
248
249         if (p != end)
250                 goto out_err;
251
252         CDEBUG(D_SEC, "successfully imported rfc1964 context\n");
253         return 0;
254 out_err:
255         return GSS_S_FAILURE;
256 }
257
258 /* Flags for version 2 context flags */
259 #define KRB5_CTX_FLAG_INITIATOR         0x00000001
260 #define KRB5_CTX_FLAG_CFX               0x00000002
261 #define KRB5_CTX_FLAG_ACCEPTOR_SUBKEY   0x00000004
262
263 static
264 __u32 import_context_rfc4121(struct krb5_ctx *kctx, char *p, char *end)
265 {
266         unsigned int    tmp_uint, keysize;
267
268         /* end time. While kc_endtime might be 64 bit the krb5 API
269          * still uses 32 bits. To delay the 2038 bug see the incoming
270          * value as a u32 which give us until 2106. See the link for details:
271          *
272          * http://web.mit.edu/kerberos/www/krb5-current/doc/appdev/y2038.html
273          */
274         if (gss_get_bytes(&p, end, &kctx->kc_endtime, sizeof(u32)))
275                 goto out_err;
276
277         /* flags */
278         if (gss_get_bytes(&p, end, &tmp_uint, sizeof(tmp_uint)))
279                 goto out_err;
280
281         if (tmp_uint & KRB5_CTX_FLAG_INITIATOR)
282                 kctx->kc_initiate = 1;
283         if (tmp_uint & KRB5_CTX_FLAG_CFX)
284                 kctx->kc_cfx = 1;
285         if (tmp_uint & KRB5_CTX_FLAG_ACCEPTOR_SUBKEY)
286                 kctx->kc_have_acceptor_subkey = 1;
287
288         /* seq send */
289         if (gss_get_bytes(&p, end, &kctx->kc_seq_send,
290             sizeof(kctx->kc_seq_send)))
291                 goto out_err;
292
293         /* enctype */
294         if (gss_get_bytes(&p, end, &kctx->kc_enctype, sizeof(kctx->kc_enctype)))
295                 goto out_err;
296
297         /* size of each key */
298         if (gss_get_bytes(&p, end, &keysize, sizeof(keysize)))
299                 goto out_err;
300
301         /* number of keys - should always be 3 */
302         if (gss_get_bytes(&p, end, &tmp_uint, sizeof(tmp_uint)))
303                 goto out_err;
304
305         if (tmp_uint != 3) {
306                 CERROR("Invalid number of keys: %u\n", tmp_uint);
307                 goto out_err;
308         }
309
310         /* ke */
311         if (gss_get_keyblock(&p, end, &kctx->kc_keye, keysize))
312                 goto out_err;
313         /* ki */
314         if (gss_get_keyblock(&p, end, &kctx->kc_keyi, keysize))
315                 goto out_err;
316         /* ki */
317         if (gss_get_keyblock(&p, end, &kctx->kc_keyc, keysize))
318                 goto out_err;
319
320         CDEBUG(D_SEC, "successfully imported v2 context\n");
321         return 0;
322 out_err:
323         return GSS_S_FAILURE;
324 }
325
326 /*
327  * The whole purpose here is trying to keep user level gss context parsing
328  * from nfs-utils unchanged as possible as we can, they are not quite mature
329  * yet, and many stuff still not clear, like heimdal etc.
330  */
331 static
332 __u32 gss_import_sec_context_kerberos(rawobj_t *inbuf,
333                                       struct gss_ctx *gctx)
334 {
335         struct krb5_ctx *kctx;
336         char *p = (char *)inbuf->data;
337         char *end = (char *)(inbuf->data + inbuf->len);
338         unsigned int tmp_uint, rc;
339
340         if (gss_get_bytes(&p, end, &tmp_uint, sizeof(tmp_uint))) {
341                 CERROR("Fail to read version\n");
342                 return GSS_S_FAILURE;
343         }
344
345         /* only support 0, 1 for the moment */
346         if (tmp_uint > 2) {
347                 CERROR("Invalid version %u\n", tmp_uint);
348                 return GSS_S_FAILURE;
349         }
350
351         OBD_ALLOC_PTR(kctx);
352         if (!kctx)
353                 return GSS_S_FAILURE;
354
355         if (tmp_uint == 0 || tmp_uint == 1) {
356                 kctx->kc_initiate = tmp_uint;
357                 rc = import_context_rfc1964(kctx, p, end);
358         } else {
359                 rc = import_context_rfc4121(kctx, p, end);
360         }
361
362         if (rc == 0)
363                 rc = krb5_init_keys(kctx);
364
365         if (rc) {
366                 delete_context_kerberos(kctx);
367                 OBD_FREE_PTR(kctx);
368
369                 return GSS_S_FAILURE;
370         }
371
372         gctx->internal_ctx_id = kctx;
373         return GSS_S_COMPLETE;
374 }
375
376 static
377 __u32 gss_copy_reverse_context_kerberos(struct gss_ctx *gctx,
378                                         struct gss_ctx *gctx_new)
379 {
380         struct krb5_ctx *kctx = gctx->internal_ctx_id;
381         struct krb5_ctx *knew;
382
383         OBD_ALLOC_PTR(knew);
384         if (!knew)
385                 return GSS_S_FAILURE;
386
387         knew->kc_initiate = kctx->kc_initiate ? 0 : 1;
388         knew->kc_cfx = kctx->kc_cfx;
389         knew->kc_seed_init = kctx->kc_seed_init;
390         knew->kc_have_acceptor_subkey = kctx->kc_have_acceptor_subkey;
391         knew->kc_endtime = kctx->kc_endtime;
392
393         memcpy(knew->kc_seed, kctx->kc_seed, sizeof(kctx->kc_seed));
394         knew->kc_seq_send = kctx->kc_seq_recv;
395         knew->kc_seq_recv = kctx->kc_seq_send;
396         knew->kc_enctype = kctx->kc_enctype;
397
398         if (rawobj_dup(&knew->kc_mech_used, &kctx->kc_mech_used))
399                 goto out_err;
400
401         if (gss_keyblock_dup(&knew->kc_keye, &kctx->kc_keye))
402                 goto out_err;
403         if (gss_keyblock_dup(&knew->kc_keyi, &kctx->kc_keyi))
404                 goto out_err;
405         if (gss_keyblock_dup(&knew->kc_keyc, &kctx->kc_keyc))
406                 goto out_err;
407         if (krb5_init_keys(knew))
408                 goto out_err;
409
410         gctx_new->internal_ctx_id = knew;
411         CDEBUG(D_SEC, "successfully copied reverse context\n");
412         return GSS_S_COMPLETE;
413
414 out_err:
415         delete_context_kerberos(knew);
416         OBD_FREE_PTR(knew);
417         return GSS_S_FAILURE;
418 }
419
420 static
421 __u32 gss_inquire_context_kerberos(struct gss_ctx *gctx,
422                                    time64_t *endtime)
423 {
424         struct krb5_ctx *kctx = gctx->internal_ctx_id;
425
426         *endtime = kctx->kc_endtime;
427         return GSS_S_COMPLETE;
428 }
429
430 static
431 void gss_delete_sec_context_kerberos(void *internal_ctx)
432 {
433         struct krb5_ctx *kctx = internal_ctx;
434
435         delete_context_kerberos(kctx);
436         OBD_FREE_PTR(kctx);
437 }
438
439 /*
440  * compute (keyed/keyless) checksum against the plain text which appended
441  * with krb5 wire token header.
442  */
443 static
444 __s32 krb5_make_checksum(__u32 enctype,
445                          struct gss_keyblock *kb,
446                          struct krb5_header *khdr,
447                          int msgcnt, rawobj_t *msgs,
448                          int iovcnt, lnet_kiov_t *iovs,
449                          rawobj_t *cksum,
450                          digest_hash hash_func)
451 {
452         struct krb5_enctype *ke = &enctypes[enctype];
453         struct ahash_request *req = NULL;
454         enum cfs_crypto_hash_alg hash_algo;
455         rawobj_t hdr;
456         int rc;
457
458         hash_algo = cfs_crypto_hash_alg(ke->ke_hash_name);
459
460         /* For the cbc(des) case we want md5 instead of hmac(md5) */
461         if (strcmp(ke->ke_enc_name, "cbc(des)"))
462                 req = cfs_crypto_hash_init(hash_algo, kb->kb_key.data,
463                                            kb->kb_key.len);
464         else
465                 req = cfs_crypto_hash_init(hash_algo, NULL, 0);
466         if (IS_ERR(req)) {
467                 rc = PTR_ERR(req);
468                 CERROR("failed to alloc hash %s : rc = %d\n",
469                        ke->ke_hash_name, rc);
470                 goto out_no_hash;
471         }
472
473         cksum->len = cfs_crypto_hash_digestsize(hash_algo);
474         OBD_ALLOC_LARGE(cksum->data, cksum->len);
475         if (!cksum->data) {
476                 cksum->len = 0;
477                 rc = -ENOMEM;
478                 goto out_free_hash;
479         }
480
481         hdr.data = (__u8 *)khdr;
482         hdr.len = sizeof(*khdr);
483
484         if (!hash_func) {
485                 rc = -EPROTO;
486                 CERROR("hash function for %s undefined\n",
487                        ke->ke_hash_name);
488                 goto out_free_hash;
489         }
490         rc = hash_func(req, &hdr, msgcnt, msgs, iovcnt, iovs);
491         if (rc)
492                 goto out_free_hash;
493
494         if (!ke->ke_hash_hmac) {
495                 LASSERT(kb->kb_tfm);
496
497                 cfs_crypto_hash_final(req, cksum->data, &cksum->len);
498                 rc = gss_crypt_generic(kb->kb_tfm, 0, NULL,
499                                        cksum->data, cksum->data,
500                                        cksum->len);
501                 goto out_no_hash;
502         }
503
504 out_free_hash:
505         if (req)
506                 cfs_crypto_hash_final(req, cksum->data, &cksum->len);
507 out_no_hash:
508         return rc ? GSS_S_FAILURE : GSS_S_COMPLETE;
509 }
510
511 static void fill_krb5_header(struct krb5_ctx *kctx,
512                              struct krb5_header *khdr,
513                              int privacy)
514 {
515         unsigned char acceptor_flag;
516
517         acceptor_flag = kctx->kc_initiate ? 0 : FLAG_SENDER_IS_ACCEPTOR;
518
519         if (privacy) {
520                 khdr->kh_tok_id = cpu_to_be16(KG_TOK_WRAP_MSG);
521                 khdr->kh_flags = acceptor_flag | FLAG_WRAP_CONFIDENTIAL;
522                 khdr->kh_ec = cpu_to_be16(0);
523                 khdr->kh_rrc = cpu_to_be16(0);
524         } else {
525                 khdr->kh_tok_id = cpu_to_be16(KG_TOK_MIC_MSG);
526                 khdr->kh_flags = acceptor_flag;
527                 khdr->kh_ec = cpu_to_be16(0xffff);
528                 khdr->kh_rrc = cpu_to_be16(0xffff);
529         }
530
531         khdr->kh_filler = 0xff;
532         spin_lock(&krb5_seq_lock);
533         khdr->kh_seq = cpu_to_be64(kctx->kc_seq_send++);
534         spin_unlock(&krb5_seq_lock);
535 }
536
537 static __u32 verify_krb5_header(struct krb5_ctx *kctx,
538                                 struct krb5_header *khdr,
539                                 int privacy)
540 {
541         unsigned char acceptor_flag;
542         __u16         tok_id, ec_rrc;
543
544         acceptor_flag = kctx->kc_initiate ? FLAG_SENDER_IS_ACCEPTOR : 0;
545
546         if (privacy) {
547                 tok_id = KG_TOK_WRAP_MSG;
548                 ec_rrc = 0x0;
549         } else {
550                 tok_id = KG_TOK_MIC_MSG;
551                 ec_rrc = 0xffff;
552         }
553
554         /* sanity checks */
555         if (be16_to_cpu(khdr->kh_tok_id) != tok_id) {
556                 CERROR("bad token id\n");
557                 return GSS_S_DEFECTIVE_TOKEN;
558         }
559         if ((khdr->kh_flags & FLAG_SENDER_IS_ACCEPTOR) != acceptor_flag) {
560                 CERROR("bad direction flag\n");
561                 return GSS_S_BAD_SIG;
562         }
563         if (privacy && (khdr->kh_flags & FLAG_WRAP_CONFIDENTIAL) == 0) {
564                 CERROR("missing confidential flag\n");
565                 return GSS_S_BAD_SIG;
566         }
567         if (khdr->kh_filler != 0xff) {
568                 CERROR("bad filler\n");
569                 return GSS_S_DEFECTIVE_TOKEN;
570         }
571         if (be16_to_cpu(khdr->kh_ec) != ec_rrc ||
572             be16_to_cpu(khdr->kh_rrc) != ec_rrc) {
573                 CERROR("bad EC or RRC\n");
574                 return GSS_S_DEFECTIVE_TOKEN;
575         }
576         return GSS_S_COMPLETE;
577 }
578
579 static
580 __u32 gss_get_mic_kerberos(struct gss_ctx *gctx,
581                            int msgcnt,
582                            rawobj_t *msgs,
583                            int iovcnt,
584                            lnet_kiov_t *iovs,
585                            rawobj_t *token)
586 {
587         struct krb5_ctx     *kctx = gctx->internal_ctx_id;
588         struct krb5_enctype *ke = &enctypes[kctx->kc_enctype];
589         struct krb5_header  *khdr;
590         rawobj_t cksum = RAWOBJ_EMPTY;
591         u32 major;
592
593         /* fill krb5 header */
594         LASSERT(token->len >= sizeof(*khdr));
595         khdr = (struct krb5_header *)token->data;
596         fill_krb5_header(kctx, khdr, 0);
597
598         /* checksum */
599         if (krb5_make_checksum(kctx->kc_enctype, &kctx->kc_keyc, khdr,
600                                msgcnt, msgs, iovcnt, iovs, &cksum,
601                                gctx->hash_func))
602                 GOTO(out_free_cksum, major = GSS_S_FAILURE);
603
604         LASSERT(cksum.len >= ke->ke_hash_size);
605         LASSERT(token->len >= sizeof(*khdr) + ke->ke_hash_size);
606         memcpy(khdr + 1, cksum.data + cksum.len - ke->ke_hash_size,
607                ke->ke_hash_size);
608
609         token->len = sizeof(*khdr) + ke->ke_hash_size;
610         major = GSS_S_COMPLETE;
611 out_free_cksum:
612         rawobj_free(&cksum);
613         return major;
614 }
615
616 static
617 __u32 gss_verify_mic_kerberos(struct gss_ctx *gctx,
618                               int msgcnt,
619                               rawobj_t *msgs,
620                               int iovcnt,
621                               lnet_kiov_t *iovs,
622                               rawobj_t *token)
623 {
624         struct krb5_ctx *kctx = gctx->internal_ctx_id;
625         struct krb5_enctype *ke = &enctypes[kctx->kc_enctype];
626         struct krb5_header *khdr;
627         rawobj_t cksum = RAWOBJ_EMPTY;
628         u32 major;
629
630         if (token->len < sizeof(*khdr)) {
631                 CERROR("short signature: %u\n", token->len);
632                 return GSS_S_DEFECTIVE_TOKEN;
633         }
634
635         khdr = (struct krb5_header *)token->data;
636
637         major = verify_krb5_header(kctx, khdr, 0);
638         if (major != GSS_S_COMPLETE) {
639                 CERROR("bad krb5 header\n");
640                 goto out;
641         }
642
643         if (token->len < sizeof(*khdr) + ke->ke_hash_size) {
644                 CERROR("short signature: %u, require %d\n",
645                        token->len, (int) sizeof(*khdr) + ke->ke_hash_size);
646                 GOTO(out, major = GSS_S_FAILURE);
647         }
648
649         if (krb5_make_checksum(kctx->kc_enctype, &kctx->kc_keyc,
650                                khdr, msgcnt, msgs, iovcnt, iovs, &cksum,
651                                gctx->hash_func))
652                 GOTO(out_free_cksum, major = GSS_S_FAILURE);
653
654         LASSERT(cksum.len >= ke->ke_hash_size);
655         if (memcmp(khdr + 1, cksum.data + cksum.len - ke->ke_hash_size,
656                    ke->ke_hash_size)) {
657                 CERROR("checksum mismatch\n");
658                 GOTO(out_free_cksum, major = GSS_S_BAD_SIG);
659         }
660         major = GSS_S_COMPLETE;
661 out_free_cksum:
662         rawobj_free(&cksum);
663 out:
664         return major;
665 }
666
667 /*
668  * if adj_nob != 0, we adjust desc->bd_nob to the actual cipher text size.
669  */
670 static
671 int krb5_encrypt_bulk(struct crypto_blkcipher *tfm,
672                       struct krb5_header *khdr,
673                       char *confounder,
674                       struct ptlrpc_bulk_desc *desc,
675                       rawobj_t *cipher,
676                       int adj_nob)
677 {
678         struct blkcipher_desc   ciph_desc;
679         __u8                    local_iv[16] = {0};
680         struct scatterlist      src, dst;
681         struct sg_table         sg_src, sg_dst;
682         int                     blocksize, i, rc, nob = 0;
683
684         LASSERT(desc->bd_iov_count);
685         LASSERT(GET_ENC_KIOV(desc));
686
687         blocksize = crypto_blkcipher_blocksize(tfm);
688         LASSERT(blocksize > 1);
689         LASSERT(cipher->len == blocksize + sizeof(*khdr));
690
691         ciph_desc.tfm  = tfm;
692         ciph_desc.info = local_iv;
693         ciph_desc.flags = 0;
694
695         /* encrypt confounder */
696         rc = gss_setup_sgtable(&sg_src, &src, confounder, blocksize);
697         if (rc != 0)
698                 return rc;
699
700         rc = gss_setup_sgtable(&sg_dst, &dst, cipher->data, blocksize);
701         if (rc != 0) {
702                 gss_teardown_sgtable(&sg_src);
703                 return rc;
704         }
705
706         rc = crypto_blkcipher_encrypt_iv(&ciph_desc, sg_dst.sgl,
707                                          sg_src.sgl, blocksize);
708
709         gss_teardown_sgtable(&sg_dst);
710         gss_teardown_sgtable(&sg_src);
711
712         if (rc) {
713                 CERROR("error to encrypt confounder: %d\n", rc);
714                 return rc;
715         }
716
717         /* encrypt clear pages */
718         for (i = 0; i < desc->bd_iov_count; i++) {
719                 sg_init_table(&src, 1);
720                 sg_set_page(&src, BD_GET_KIOV(desc, i).kiov_page,
721                             (BD_GET_KIOV(desc, i).kiov_len +
722                                 blocksize - 1) &
723                             (~(blocksize - 1)),
724                             BD_GET_KIOV(desc, i).kiov_offset);
725                 if (adj_nob)
726                         nob += src.length;
727                 sg_init_table(&dst, 1);
728                 sg_set_page(&dst, BD_GET_ENC_KIOV(desc, i).kiov_page,
729                             src.length, src.offset);
730
731                 BD_GET_ENC_KIOV(desc, i).kiov_offset = dst.offset;
732                 BD_GET_ENC_KIOV(desc, i).kiov_len = dst.length;
733
734                 rc = crypto_blkcipher_encrypt_iv(&ciph_desc, &dst, &src,
735                                                     src.length);
736                 if (rc) {
737                         CERROR("error to encrypt page: %d\n", rc);
738                         return rc;
739                 }
740         }
741
742         /* encrypt krb5 header */
743         rc = gss_setup_sgtable(&sg_src, &src, khdr, sizeof(*khdr));
744         if (rc != 0)
745                 return rc;
746
747         rc = gss_setup_sgtable(&sg_dst, &dst, cipher->data + blocksize,
748                            sizeof(*khdr));
749         if (rc != 0) {
750                 gss_teardown_sgtable(&sg_src);
751                 return rc;
752         }
753
754         rc = crypto_blkcipher_encrypt_iv(&ciph_desc, sg_dst.sgl, sg_src.sgl,
755                                          sizeof(*khdr));
756
757         gss_teardown_sgtable(&sg_dst);
758         gss_teardown_sgtable(&sg_src);
759
760         if (rc) {
761                 CERROR("error to encrypt krb5 header: %d\n", rc);
762                 return rc;
763         }
764
765         if (adj_nob)
766                 desc->bd_nob = nob;
767
768         return 0;
769 }
770
771 /*
772  * desc->bd_nob_transferred is the size of cipher text received.
773  * desc->bd_nob is the target size of plain text supposed to be.
774  *
775  * if adj_nob != 0, we adjust each page's kiov_len to the actual
776  * plain text size.
777  * - for client read: we don't know data size for each page, so
778  *   bd_iov[]->kiov_len is set to PAGE_SIZE, but actual data received might
779  *   be smaller, so we need to adjust it according to
780  *   bd_u.bd_kiov.bd_enc_vec[]->kiov_len.
781  *   this means we DO NOT support the situation that server send an odd size
782  *   data in a page which is not the last one.
783  * - for server write: we knows exactly data size for each page being expected,
784  *   thus kiov_len is accurate already, so we should not adjust it at all.
785  *   and bd_u.bd_kiov.bd_enc_vec[]->kiov_len should be
786  *   round_up(bd_iov[]->kiov_len) which
787  *   should have been done by prep_bulk().
788  */
789 static
790 int krb5_decrypt_bulk(struct crypto_blkcipher *tfm,
791                       struct krb5_header *khdr,
792                       struct ptlrpc_bulk_desc *desc,
793                       rawobj_t *cipher,
794                       rawobj_t *plain,
795                       int adj_nob)
796 {
797         struct blkcipher_desc   ciph_desc;
798         __u8                    local_iv[16] = {0};
799         struct scatterlist      src, dst;
800         struct sg_table         sg_src, sg_dst;
801         int                     ct_nob = 0, pt_nob = 0;
802         int                     blocksize, i, rc;
803
804         LASSERT(desc->bd_iov_count);
805         LASSERT(GET_ENC_KIOV(desc));
806         LASSERT(desc->bd_nob_transferred);
807
808         blocksize = crypto_blkcipher_blocksize(tfm);
809         LASSERT(blocksize > 1);
810         LASSERT(cipher->len == blocksize + sizeof(*khdr));
811
812         ciph_desc.tfm  = tfm;
813         ciph_desc.info = local_iv;
814         ciph_desc.flags = 0;
815
816         if (desc->bd_nob_transferred % blocksize) {
817                 CERROR("odd transferred nob: %d\n", desc->bd_nob_transferred);
818                 return -EPROTO;
819         }
820
821         /* decrypt head (confounder) */
822         rc = gss_setup_sgtable(&sg_src, &src, cipher->data, blocksize);
823         if (rc != 0)
824                 return rc;
825
826         rc = gss_setup_sgtable(&sg_dst, &dst, plain->data, blocksize);
827         if (rc != 0) {
828                 gss_teardown_sgtable(&sg_src);
829                 return rc;
830         }
831
832         rc = crypto_blkcipher_decrypt_iv(&ciph_desc, sg_dst.sgl,
833                                          sg_src.sgl, blocksize);
834
835         gss_teardown_sgtable(&sg_dst);
836         gss_teardown_sgtable(&sg_src);
837
838         if (rc) {
839                 CERROR("error to decrypt confounder: %d\n", rc);
840                 return rc;
841         }
842
843         for (i = 0; i < desc->bd_iov_count && ct_nob < desc->bd_nob_transferred;
844              i++) {
845                 if (BD_GET_ENC_KIOV(desc, i).kiov_offset % blocksize
846                     != 0 ||
847                     BD_GET_ENC_KIOV(desc, i).kiov_len % blocksize
848                     != 0) {
849                         CERROR("page %d: odd offset %u len %u, blocksize %d\n",
850                                i, BD_GET_ENC_KIOV(desc, i).kiov_offset,
851                                BD_GET_ENC_KIOV(desc, i).kiov_len,
852                                blocksize);
853                         return -EFAULT;
854                 }
855
856                 if (adj_nob) {
857                         if (ct_nob + BD_GET_ENC_KIOV(desc, i).kiov_len >
858                             desc->bd_nob_transferred)
859                                 BD_GET_ENC_KIOV(desc, i).kiov_len =
860                                         desc->bd_nob_transferred - ct_nob;
861
862                         BD_GET_KIOV(desc, i).kiov_len =
863                           BD_GET_ENC_KIOV(desc, i).kiov_len;
864                         if (pt_nob + BD_GET_ENC_KIOV(desc, i).kiov_len >
865                             desc->bd_nob)
866                                 BD_GET_KIOV(desc, i).kiov_len =
867                                   desc->bd_nob - pt_nob;
868                 } else {
869                         /* this should be guaranteed by LNET */
870                         LASSERT(ct_nob + BD_GET_ENC_KIOV(desc, i).
871                                 kiov_len <=
872                                 desc->bd_nob_transferred);
873                         LASSERT(BD_GET_KIOV(desc, i).kiov_len <=
874                                 BD_GET_ENC_KIOV(desc, i).kiov_len);
875                 }
876
877                 if (BD_GET_ENC_KIOV(desc, i).kiov_len == 0)
878                         continue;
879
880                 sg_init_table(&src, 1);
881                 sg_set_page(&src, BD_GET_ENC_KIOV(desc, i).kiov_page,
882                             BD_GET_ENC_KIOV(desc, i).kiov_len,
883                             BD_GET_ENC_KIOV(desc, i).kiov_offset);
884                 dst = src;
885                 if (BD_GET_KIOV(desc, i).kiov_len % blocksize == 0)
886                         sg_assign_page(&dst,
887                                        BD_GET_KIOV(desc, i).kiov_page);
888
889                 rc = crypto_blkcipher_decrypt_iv(&ciph_desc, &dst, &src,
890                                                  src.length);
891                 if (rc) {
892                         CERROR("error to decrypt page: %d\n", rc);
893                         return rc;
894                 }
895
896                 if (BD_GET_KIOV(desc, i).kiov_len % blocksize != 0) {
897                         memcpy(page_address(BD_GET_KIOV(desc, i).kiov_page) +
898                                BD_GET_KIOV(desc, i).kiov_offset,
899                                page_address(BD_GET_ENC_KIOV(desc, i).
900                                             kiov_page) +
901                                BD_GET_KIOV(desc, i).kiov_offset,
902                                BD_GET_KIOV(desc, i).kiov_len);
903                 }
904
905                 ct_nob += BD_GET_ENC_KIOV(desc, i).kiov_len;
906                 pt_nob += BD_GET_KIOV(desc, i).kiov_len;
907         }
908
909         if (unlikely(ct_nob != desc->bd_nob_transferred)) {
910                 CERROR("%d cipher text transferred but only %d decrypted\n",
911                        desc->bd_nob_transferred, ct_nob);
912                 return -EFAULT;
913         }
914
915         if (unlikely(!adj_nob && pt_nob != desc->bd_nob)) {
916                 CERROR("%d plain text expected but only %d received\n",
917                        desc->bd_nob, pt_nob);
918                 return -EFAULT;
919         }
920
921         /* if needed, clear up the rest unused iovs */
922         if (adj_nob)
923                 while (i < desc->bd_iov_count)
924                         BD_GET_KIOV(desc, i++).kiov_len = 0;
925
926         /* decrypt tail (krb5 header) */
927         rc = gss_setup_sgtable(&sg_src, &src, cipher->data + blocksize,
928                                sizeof(*khdr));
929         if (rc != 0)
930                 return rc;
931
932         rc = gss_setup_sgtable(&sg_dst, &dst, cipher->data + blocksize,
933                                sizeof(*khdr));
934         if (rc != 0) {
935                 gss_teardown_sgtable(&sg_src);
936                 return rc;
937         }
938
939         rc = crypto_blkcipher_decrypt_iv(&ciph_desc, sg_dst.sgl, sg_src.sgl,
940                                          sizeof(*khdr));
941
942         gss_teardown_sgtable(&sg_src);
943         gss_teardown_sgtable(&sg_dst);
944
945         if (rc) {
946                 CERROR("error to decrypt tail: %d\n", rc);
947                 return rc;
948         }
949
950         if (memcmp(cipher->data + blocksize, khdr, sizeof(*khdr))) {
951                 CERROR("krb5 header doesn't match\n");
952                 return -EACCES;
953         }
954
955         return 0;
956 }
957
958 static
959 __u32 gss_wrap_kerberos(struct gss_ctx *gctx,
960                         rawobj_t *gsshdr,
961                         rawobj_t *msg,
962                         int msg_buflen,
963                         rawobj_t *token)
964 {
965         struct krb5_ctx     *kctx = gctx->internal_ctx_id;
966         struct krb5_enctype *ke = &enctypes[kctx->kc_enctype];
967         struct krb5_header  *khdr;
968         int                  blocksize;
969         rawobj_t             cksum = RAWOBJ_EMPTY;
970         rawobj_t             data_desc[3], cipher;
971         __u8                 conf[GSS_MAX_CIPHER_BLOCK];
972         __u8                 local_iv[16] = {0};
973         u32 major;
974         int                  rc = 0;
975
976         LASSERT(ke);
977         LASSERT(ke->ke_conf_size <= GSS_MAX_CIPHER_BLOCK);
978         LASSERT(kctx->kc_keye.kb_tfm == NULL ||
979                 ke->ke_conf_size >=
980                 crypto_blkcipher_blocksize(kctx->kc_keye.kb_tfm));
981
982         /*
983          * final token format:
984          * ---------------------------------------------------
985          * | krb5 header | cipher text | checksum (16 bytes) |
986          * ---------------------------------------------------
987          */
988
989         /* fill krb5 header */
990         LASSERT(token->len >= sizeof(*khdr));
991         khdr = (struct krb5_header *)token->data;
992         fill_krb5_header(kctx, khdr, 1);
993
994         /* generate confounder */
995         get_random_bytes(conf, ke->ke_conf_size);
996
997         /* get encryption blocksize. note kc_keye might not associated with
998          * a tfm, currently only for arcfour-hmac */
999         if (kctx->kc_enctype == ENCTYPE_ARCFOUR_HMAC) {
1000                 LASSERT(kctx->kc_keye.kb_tfm == NULL);
1001                 blocksize = 1;
1002         } else {
1003                 LASSERT(kctx->kc_keye.kb_tfm);
1004                 blocksize = crypto_blkcipher_blocksize(kctx->kc_keye.kb_tfm);
1005         }
1006         LASSERT(blocksize <= ke->ke_conf_size);
1007
1008         /* padding the message */
1009         if (gss_add_padding(msg, msg_buflen, blocksize))
1010                 return GSS_S_FAILURE;
1011
1012         /*
1013          * clear text layout for checksum:
1014          * ------------------------------------------------------
1015          * | confounder | gss header | clear msgs | krb5 header |
1016          * ------------------------------------------------------
1017          */
1018         data_desc[0].data = conf;
1019         data_desc[0].len = ke->ke_conf_size;
1020         data_desc[1].data = gsshdr->data;
1021         data_desc[1].len = gsshdr->len;
1022         data_desc[2].data = msg->data;
1023         data_desc[2].len = msg->len;
1024
1025         /* compute checksum */
1026         if (krb5_make_checksum(kctx->kc_enctype, &kctx->kc_keyi,
1027                                khdr, 3, data_desc, 0, NULL, &cksum,
1028                                gctx->hash_func))
1029                 GOTO(out_free_cksum, major = GSS_S_FAILURE);
1030         LASSERT(cksum.len >= ke->ke_hash_size);
1031
1032         /*
1033          * clear text layout for encryption:
1034          * -----------------------------------------
1035          * | confounder | clear msgs | krb5 header |
1036          * -----------------------------------------
1037          */
1038         data_desc[0].data = conf;
1039         data_desc[0].len = ke->ke_conf_size;
1040         data_desc[1].data = msg->data;
1041         data_desc[1].len = msg->len;
1042         data_desc[2].data = (__u8 *) khdr;
1043         data_desc[2].len = sizeof(*khdr);
1044
1045         /* cipher text will be directly inplace */
1046         cipher.data = (__u8 *)(khdr + 1);
1047         cipher.len = token->len - sizeof(*khdr);
1048         LASSERT(cipher.len >= ke->ke_conf_size + msg->len + sizeof(*khdr));
1049
1050         if (kctx->kc_enctype == ENCTYPE_ARCFOUR_HMAC) {
1051                 rawobj_t arc4_keye = RAWOBJ_EMPTY;
1052                 struct crypto_blkcipher *arc4_tfm;
1053
1054                 if (krb5_make_checksum(ENCTYPE_ARCFOUR_HMAC, &kctx->kc_keyi,
1055                                        NULL, 1, &cksum, 0, NULL, &arc4_keye,
1056                                        gctx->hash_func)) {
1057                         CERROR("failed to obtain arc4 enc key\n");
1058                         GOTO(arc4_out_key, rc = -EACCES);
1059                 }
1060
1061                 arc4_tfm = crypto_alloc_blkcipher("ecb(arc4)", 0, 0);
1062                 if (IS_ERR(arc4_tfm)) {
1063                         CERROR("failed to alloc tfm arc4 in ECB mode\n");
1064                         GOTO(arc4_out_key, rc = -EACCES);
1065                 }
1066
1067                 if (crypto_blkcipher_setkey(arc4_tfm, arc4_keye.data,
1068                                             arc4_keye.len)) {
1069                         CERROR("failed to set arc4 key, len %d\n",
1070                                arc4_keye.len);
1071                         GOTO(arc4_out_tfm, rc = -EACCES);
1072                 }
1073
1074                 rc = gss_crypt_rawobjs(arc4_tfm, NULL, 3, data_desc,
1075                                        &cipher, 1);
1076 arc4_out_tfm:
1077                 crypto_free_blkcipher(arc4_tfm);
1078 arc4_out_key:
1079                 rawobj_free(&arc4_keye);
1080         } else {
1081                 rc = gss_crypt_rawobjs(kctx->kc_keye.kb_tfm, local_iv, 3,
1082                                        data_desc, &cipher, 1);
1083         }
1084
1085         if (rc)
1086                 GOTO(out_free_cksum, major = GSS_S_FAILURE);
1087
1088         /* fill in checksum */
1089         LASSERT(token->len >= sizeof(*khdr) + cipher.len + ke->ke_hash_size);
1090         memcpy((char *)(khdr + 1) + cipher.len,
1091                cksum.data + cksum.len - ke->ke_hash_size,
1092                ke->ke_hash_size);
1093
1094         /* final token length */
1095         token->len = sizeof(*khdr) + cipher.len + ke->ke_hash_size;
1096         major = GSS_S_COMPLETE;
1097 out_free_cksum:
1098         rawobj_free(&cksum);
1099         return major;
1100 }
1101
1102 static
1103 __u32 gss_prep_bulk_kerberos(struct gss_ctx *gctx,
1104                              struct ptlrpc_bulk_desc *desc)
1105 {
1106         struct krb5_ctx     *kctx = gctx->internal_ctx_id;
1107         int                  blocksize, i;
1108
1109         LASSERT(desc->bd_iov_count);
1110         LASSERT(GET_ENC_KIOV(desc));
1111         LASSERT(kctx->kc_keye.kb_tfm);
1112
1113         blocksize = crypto_blkcipher_blocksize(kctx->kc_keye.kb_tfm);
1114
1115         for (i = 0; i < desc->bd_iov_count; i++) {
1116                 LASSERT(BD_GET_ENC_KIOV(desc, i).kiov_page);
1117                 /*
1118                  * offset should always start at page boundary of either
1119                  * client or server side.
1120                  */
1121                 if (BD_GET_KIOV(desc, i).kiov_offset & blocksize) {
1122                         CERROR("odd offset %d in page %d\n",
1123                                BD_GET_KIOV(desc, i).kiov_offset, i);
1124                         return GSS_S_FAILURE;
1125                 }
1126
1127                 BD_GET_ENC_KIOV(desc, i).kiov_offset =
1128                         BD_GET_KIOV(desc, i).kiov_offset;
1129                 BD_GET_ENC_KIOV(desc, i).kiov_len =
1130                         (BD_GET_KIOV(desc, i).kiov_len +
1131                          blocksize - 1) & (~(blocksize - 1));
1132         }
1133
1134         return GSS_S_COMPLETE;
1135 }
1136
1137 static
1138 __u32 gss_wrap_bulk_kerberos(struct gss_ctx *gctx,
1139                              struct ptlrpc_bulk_desc *desc,
1140                              rawobj_t *token, int adj_nob)
1141 {
1142         struct krb5_ctx     *kctx = gctx->internal_ctx_id;
1143         struct krb5_enctype *ke = &enctypes[kctx->kc_enctype];
1144         struct krb5_header  *khdr;
1145         int                  blocksize;
1146         rawobj_t             cksum = RAWOBJ_EMPTY;
1147         rawobj_t             data_desc[1], cipher;
1148         __u8                 conf[GSS_MAX_CIPHER_BLOCK];
1149         int rc = 0;
1150         u32 major;
1151
1152         LASSERT(ke);
1153         LASSERT(ke->ke_conf_size <= GSS_MAX_CIPHER_BLOCK);
1154
1155         /*
1156          * final token format:
1157          * --------------------------------------------------
1158          * | krb5 header | head/tail cipher text | checksum |
1159          * --------------------------------------------------
1160          */
1161
1162         /* fill krb5 header */
1163         LASSERT(token->len >= sizeof(*khdr));
1164         khdr = (struct krb5_header *)token->data;
1165         fill_krb5_header(kctx, khdr, 1);
1166
1167         /* generate confounder */
1168         get_random_bytes(conf, ke->ke_conf_size);
1169
1170         /* get encryption blocksize. note kc_keye might not associated with
1171          * a tfm, currently only for arcfour-hmac */
1172         if (kctx->kc_enctype == ENCTYPE_ARCFOUR_HMAC) {
1173                 LASSERT(kctx->kc_keye.kb_tfm == NULL);
1174                 blocksize = 1;
1175         } else {
1176                 LASSERT(kctx->kc_keye.kb_tfm);
1177                 blocksize = crypto_blkcipher_blocksize(kctx->kc_keye.kb_tfm);
1178         }
1179
1180         /*
1181          * we assume the size of krb5_header (16 bytes) must be n * blocksize.
1182          * the bulk token size would be exactly (sizeof(krb5_header) +
1183          * blocksize + sizeof(krb5_header) + hashsize)
1184          */
1185         LASSERT(blocksize <= ke->ke_conf_size);
1186         LASSERT(sizeof(*khdr) >= blocksize && sizeof(*khdr) % blocksize == 0);
1187         LASSERT(token->len >= sizeof(*khdr) + blocksize + sizeof(*khdr) + 16);
1188
1189         /*
1190          * clear text layout for checksum:
1191          * ------------------------------------------
1192          * | confounder | clear pages | krb5 header |
1193          * ------------------------------------------
1194          */
1195         data_desc[0].data = conf;
1196         data_desc[0].len = ke->ke_conf_size;
1197
1198         /* compute checksum */
1199         if (krb5_make_checksum(kctx->kc_enctype, &kctx->kc_keyi,
1200                                khdr, 1, data_desc,
1201                                desc->bd_iov_count, GET_KIOV(desc),
1202                                &cksum, gctx->hash_func))
1203                 GOTO(out_free_cksum, major = GSS_S_FAILURE);
1204         LASSERT(cksum.len >= ke->ke_hash_size);
1205
1206         /*
1207          * clear text layout for encryption:
1208          * ------------------------------------------
1209          * | confounder | clear pages | krb5 header |
1210          * ------------------------------------------
1211          *        |              |             |
1212          *        ----------  (cipher pages)   |
1213          * result token:   |                   |
1214          * -------------------------------------------
1215          * | krb5 header | cipher text | cipher text |
1216          * -------------------------------------------
1217          */
1218         data_desc[0].data = conf;
1219         data_desc[0].len = ke->ke_conf_size;
1220
1221         cipher.data = (__u8 *)(khdr + 1);
1222         cipher.len = blocksize + sizeof(*khdr);
1223
1224         if (kctx->kc_enctype == ENCTYPE_ARCFOUR_HMAC) {
1225                 LBUG();
1226                 rc = 0;
1227         } else {
1228                 rc = krb5_encrypt_bulk(kctx->kc_keye.kb_tfm, khdr,
1229                                        conf, desc, &cipher, adj_nob);
1230         }
1231         if (rc)
1232                 GOTO(out_free_cksum, major = GSS_S_FAILURE);
1233
1234         /* fill in checksum */
1235         LASSERT(token->len >= sizeof(*khdr) + cipher.len + ke->ke_hash_size);
1236         memcpy((char *)(khdr + 1) + cipher.len,
1237                cksum.data + cksum.len - ke->ke_hash_size,
1238                ke->ke_hash_size);
1239
1240         /* final token length */
1241         token->len = sizeof(*khdr) + cipher.len + ke->ke_hash_size;
1242         major = GSS_S_COMPLETE;
1243 out_free_cksum:
1244         rawobj_free(&cksum);
1245         return major;
1246 }
1247
1248 static
1249 __u32 gss_unwrap_kerberos(struct gss_ctx  *gctx,
1250                           rawobj_t        *gsshdr,
1251                           rawobj_t        *token,
1252                           rawobj_t        *msg)
1253 {
1254         struct krb5_ctx     *kctx = gctx->internal_ctx_id;
1255         struct krb5_enctype *ke = &enctypes[kctx->kc_enctype];
1256         struct krb5_header  *khdr;
1257         unsigned char       *tmpbuf;
1258         int                  blocksize, bodysize;
1259         rawobj_t             cksum = RAWOBJ_EMPTY;
1260         rawobj_t             cipher_in, plain_out;
1261         rawobj_t             hash_objs[3];
1262         int                  rc = 0;
1263         __u32                major;
1264         __u8                 local_iv[16] = {0};
1265
1266         LASSERT(ke);
1267
1268         if (token->len < sizeof(*khdr)) {
1269                 CERROR("short signature: %u\n", token->len);
1270                 return GSS_S_DEFECTIVE_TOKEN;
1271         }
1272
1273         khdr = (struct krb5_header *)token->data;
1274
1275         major = verify_krb5_header(kctx, khdr, 1);
1276         if (major != GSS_S_COMPLETE) {
1277                 CERROR("bad krb5 header\n");
1278                 return major;
1279         }
1280
1281         /* block size */
1282         if (kctx->kc_enctype == ENCTYPE_ARCFOUR_HMAC) {
1283                 LASSERT(kctx->kc_keye.kb_tfm == NULL);
1284                 blocksize = 1;
1285         } else {
1286                 LASSERT(kctx->kc_keye.kb_tfm);
1287                 blocksize = crypto_blkcipher_blocksize(kctx->kc_keye.kb_tfm);
1288         }
1289
1290         /* expected token layout:
1291          * ----------------------------------------
1292          * | krb5 header | cipher text | checksum |
1293          * ----------------------------------------
1294          */
1295         bodysize = token->len - sizeof(*khdr) - ke->ke_hash_size;
1296
1297         if (bodysize % blocksize) {
1298                 CERROR("odd bodysize %d\n", bodysize);
1299                 return GSS_S_DEFECTIVE_TOKEN;
1300         }
1301
1302         if (bodysize <= ke->ke_conf_size + sizeof(*khdr)) {
1303                 CERROR("incomplete token: bodysize %d\n", bodysize);
1304                 return GSS_S_DEFECTIVE_TOKEN;
1305         }
1306
1307         if (msg->len < bodysize - ke->ke_conf_size - sizeof(*khdr)) {
1308                 CERROR("buffer too small: %u, require %d\n",
1309                        msg->len, bodysize - ke->ke_conf_size);
1310                 return GSS_S_FAILURE;
1311         }
1312
1313         /* decrypting */
1314         OBD_ALLOC_LARGE(tmpbuf, bodysize);
1315         if (!tmpbuf)
1316                 return GSS_S_FAILURE;
1317
1318         major = GSS_S_FAILURE;
1319
1320         cipher_in.data = (__u8 *)(khdr + 1);
1321         cipher_in.len = bodysize;
1322         plain_out.data = tmpbuf;
1323         plain_out.len = bodysize;
1324
1325         if (kctx->kc_enctype == ENCTYPE_ARCFOUR_HMAC) {
1326                 rawobj_t                 arc4_keye;
1327                 struct crypto_blkcipher *arc4_tfm;
1328
1329                 cksum.data = token->data + token->len - ke->ke_hash_size;
1330                 cksum.len = ke->ke_hash_size;
1331
1332                 if (krb5_make_checksum(ENCTYPE_ARCFOUR_HMAC, &kctx->kc_keyi,
1333                                        NULL, 1, &cksum, 0, NULL, &arc4_keye,
1334                                        gctx->hash_func)) {
1335                         CERROR("failed to obtain arc4 enc key\n");
1336                         GOTO(arc4_out, rc = -EACCES);
1337                 }
1338
1339                 arc4_tfm = crypto_alloc_blkcipher("ecb(arc4)", 0, 0);
1340                 if (IS_ERR(arc4_tfm)) {
1341                         CERROR("failed to alloc tfm arc4 in ECB mode\n");
1342                         GOTO(arc4_out_key, rc = -EACCES);
1343                 }
1344
1345                 if (crypto_blkcipher_setkey(arc4_tfm,
1346                                             arc4_keye.data, arc4_keye.len)) {
1347                         CERROR("failed to set arc4 key, len %d\n",
1348                                arc4_keye.len);
1349                         GOTO(arc4_out_tfm, rc = -EACCES);
1350                 }
1351
1352                 rc = gss_crypt_rawobjs(arc4_tfm, NULL, 1, &cipher_in,
1353                                        &plain_out, 0);
1354 arc4_out_tfm:
1355                 crypto_free_blkcipher(arc4_tfm);
1356 arc4_out_key:
1357                 rawobj_free(&arc4_keye);
1358 arc4_out:
1359                 cksum = RAWOBJ_EMPTY;
1360         } else {
1361                 rc = gss_crypt_rawobjs(kctx->kc_keye.kb_tfm, local_iv, 1,
1362                                        &cipher_in, &plain_out, 0);
1363         }
1364
1365         if (rc != 0) {
1366                 CERROR("error decrypt\n");
1367                 goto out_free;
1368         }
1369         LASSERT(plain_out.len == bodysize);
1370
1371         /* expected clear text layout:
1372          * -----------------------------------------
1373          * | confounder | clear msgs | krb5 header |
1374          * -----------------------------------------
1375          */
1376
1377         /* verify krb5 header in token is not modified */
1378         if (memcmp(khdr, plain_out.data + plain_out.len - sizeof(*khdr),
1379                    sizeof(*khdr))) {
1380                 CERROR("decrypted krb5 header mismatch\n");
1381                 goto out_free;
1382         }
1383
1384         /* verify checksum, compose clear text as layout:
1385          * ------------------------------------------------------
1386          * | confounder | gss header | clear msgs | krb5 header |
1387          * ------------------------------------------------------
1388          */
1389         hash_objs[0].len = ke->ke_conf_size;
1390         hash_objs[0].data = plain_out.data;
1391         hash_objs[1].len = gsshdr->len;
1392         hash_objs[1].data = gsshdr->data;
1393         hash_objs[2].len = plain_out.len - ke->ke_conf_size - sizeof(*khdr);
1394         hash_objs[2].data = plain_out.data + ke->ke_conf_size;
1395         if (krb5_make_checksum(kctx->kc_enctype, &kctx->kc_keyi,
1396                                khdr, 3, hash_objs, 0, NULL, &cksum,
1397                                gctx->hash_func))
1398                 goto out_free;
1399
1400         LASSERT(cksum.len >= ke->ke_hash_size);
1401         if (memcmp((char *)(khdr + 1) + bodysize,
1402                    cksum.data + cksum.len - ke->ke_hash_size,
1403                    ke->ke_hash_size)) {
1404                 CERROR("checksum mismatch\n");
1405                 goto out_free;
1406         }
1407
1408         msg->len =  bodysize - ke->ke_conf_size - sizeof(*khdr);
1409         memcpy(msg->data, tmpbuf + ke->ke_conf_size, msg->len);
1410
1411         major = GSS_S_COMPLETE;
1412 out_free:
1413         OBD_FREE_LARGE(tmpbuf, bodysize);
1414         rawobj_free(&cksum);
1415         return major;
1416 }
1417
1418 static
1419 __u32 gss_unwrap_bulk_kerberos(struct gss_ctx *gctx,
1420                                struct ptlrpc_bulk_desc *desc,
1421                                rawobj_t *token, int adj_nob)
1422 {
1423         struct krb5_ctx     *kctx = gctx->internal_ctx_id;
1424         struct krb5_enctype *ke = &enctypes[kctx->kc_enctype];
1425         struct krb5_header  *khdr;
1426         int                  blocksize;
1427         rawobj_t             cksum = RAWOBJ_EMPTY;
1428         rawobj_t             cipher, plain;
1429         rawobj_t             data_desc[1];
1430         int                  rc;
1431         __u32                major;
1432
1433         LASSERT(ke);
1434
1435         if (token->len < sizeof(*khdr)) {
1436                 CERROR("short signature: %u\n", token->len);
1437                 return GSS_S_DEFECTIVE_TOKEN;
1438         }
1439
1440         khdr = (struct krb5_header *)token->data;
1441
1442         major = verify_krb5_header(kctx, khdr, 1);
1443         if (major != GSS_S_COMPLETE) {
1444                 CERROR("bad krb5 header\n");
1445                 return major;
1446         }
1447
1448         /* block size */
1449         if (kctx->kc_enctype == ENCTYPE_ARCFOUR_HMAC) {
1450                 LASSERT(kctx->kc_keye.kb_tfm == NULL);
1451                 blocksize = 1;
1452                 LBUG();
1453         } else {
1454                 LASSERT(kctx->kc_keye.kb_tfm);
1455                 blocksize = crypto_blkcipher_blocksize(kctx->kc_keye.kb_tfm);
1456         }
1457         LASSERT(sizeof(*khdr) >= blocksize && sizeof(*khdr) % blocksize == 0);
1458
1459         /*
1460          * token format is expected as:
1461          * -----------------------------------------------
1462          * | krb5 header | head/tail cipher text | cksum |
1463          * -----------------------------------------------
1464          */
1465         if (token->len < sizeof(*khdr) + blocksize + sizeof(*khdr) +
1466             ke->ke_hash_size) {
1467                 CERROR("short token size: %u\n", token->len);
1468                 return GSS_S_DEFECTIVE_TOKEN;
1469         }
1470
1471         cipher.data = (__u8 *) (khdr + 1);
1472         cipher.len = blocksize + sizeof(*khdr);
1473         plain.data = cipher.data;
1474         plain.len = cipher.len;
1475
1476         rc = krb5_decrypt_bulk(kctx->kc_keye.kb_tfm, khdr,
1477                                desc, &cipher, &plain, adj_nob);
1478         if (rc)
1479                 return GSS_S_DEFECTIVE_TOKEN;
1480
1481         /*
1482          * verify checksum, compose clear text as layout:
1483          * ------------------------------------------
1484          * | confounder | clear pages | krb5 header |
1485          * ------------------------------------------
1486          */
1487         data_desc[0].data = plain.data;
1488         data_desc[0].len = blocksize;
1489
1490         if (krb5_make_checksum(kctx->kc_enctype, &kctx->kc_keyi,
1491                                khdr, 1, data_desc,
1492                                desc->bd_iov_count,
1493                                GET_KIOV(desc),
1494                                &cksum, gctx->hash_func))
1495                 return GSS_S_FAILURE;
1496         LASSERT(cksum.len >= ke->ke_hash_size);
1497
1498         if (memcmp(plain.data + blocksize + sizeof(*khdr),
1499                    cksum.data + cksum.len - ke->ke_hash_size,
1500                    ke->ke_hash_size)) {
1501                 CERROR("checksum mismatch\n");
1502                 rawobj_free(&cksum);
1503                 return GSS_S_BAD_SIG;
1504         }
1505
1506         rawobj_free(&cksum);
1507         return GSS_S_COMPLETE;
1508 }
1509
1510 int gss_display_kerberos(struct gss_ctx        *ctx,
1511                          char                  *buf,
1512                          int                    bufsize)
1513 {
1514         struct krb5_ctx    *kctx = ctx->internal_ctx_id;
1515         int                 written;
1516
1517         written = snprintf(buf, bufsize, "krb5 (%s)",
1518                            enctype2str(kctx->kc_enctype));
1519         return written;
1520 }
1521
1522 static struct gss_api_ops gss_kerberos_ops = {
1523         .gss_import_sec_context     = gss_import_sec_context_kerberos,
1524         .gss_copy_reverse_context   = gss_copy_reverse_context_kerberos,
1525         .gss_inquire_context        = gss_inquire_context_kerberos,
1526         .gss_get_mic                = gss_get_mic_kerberos,
1527         .gss_verify_mic             = gss_verify_mic_kerberos,
1528         .gss_wrap                   = gss_wrap_kerberos,
1529         .gss_unwrap                 = gss_unwrap_kerberos,
1530         .gss_prep_bulk              = gss_prep_bulk_kerberos,
1531         .gss_wrap_bulk              = gss_wrap_bulk_kerberos,
1532         .gss_unwrap_bulk            = gss_unwrap_bulk_kerberos,
1533         .gss_delete_sec_context     = gss_delete_sec_context_kerberos,
1534         .gss_display                = gss_display_kerberos,
1535 };
1536
1537 static struct subflavor_desc gss_kerberos_sfs[] = {
1538         {
1539                 .sf_subflavor   = SPTLRPC_SUBFLVR_KRB5N,
1540                 .sf_qop         = 0,
1541                 .sf_service     = SPTLRPC_SVC_NULL,
1542                 .sf_name        = "krb5n"
1543         },
1544         {
1545                 .sf_subflavor   = SPTLRPC_SUBFLVR_KRB5A,
1546                 .sf_qop         = 0,
1547                 .sf_service     = SPTLRPC_SVC_AUTH,
1548                 .sf_name        = "krb5a"
1549         },
1550         {
1551                 .sf_subflavor   = SPTLRPC_SUBFLVR_KRB5I,
1552                 .sf_qop         = 0,
1553                 .sf_service     = SPTLRPC_SVC_INTG,
1554                 .sf_name        = "krb5i"
1555         },
1556         {
1557                 .sf_subflavor   = SPTLRPC_SUBFLVR_KRB5P,
1558                 .sf_qop         = 0,
1559                 .sf_service     = SPTLRPC_SVC_PRIV,
1560                 .sf_name        = "krb5p"
1561         },
1562 };
1563
1564 static struct gss_api_mech gss_kerberos_mech = {
1565         /* .gm_owner uses default NULL value for THIS_MODULE */
1566         .gm_name        = "krb5",
1567         .gm_oid         = (rawobj_t)
1568                                 {9, "\052\206\110\206\367\022\001\002\002"},
1569         .gm_ops         = &gss_kerberos_ops,
1570         .gm_sf_num      = 4,
1571         .gm_sfs         = gss_kerberos_sfs,
1572 };
1573
1574 int __init init_kerberos_module(void)
1575 {
1576         int status;
1577
1578         status = lgss_mech_register(&gss_kerberos_mech);
1579         if (status)
1580                 CERROR("Failed to register kerberos gss mechanism!\n");
1581         return status;
1582 }
1583
1584 void cleanup_kerberos_module(void)
1585 {
1586         lgss_mech_unregister(&gss_kerberos_mech);
1587 }