1*4882a593Smuzhiyun /*
2*4882a593Smuzhiyun * linux/net/sunrpc/gss_krb5_crypto.c
3*4882a593Smuzhiyun *
4*4882a593Smuzhiyun * Copyright (c) 2000-2008 The Regents of the University of Michigan.
5*4882a593Smuzhiyun * All rights reserved.
6*4882a593Smuzhiyun *
7*4882a593Smuzhiyun * Andy Adamson <andros@umich.edu>
8*4882a593Smuzhiyun * Bruce Fields <bfields@umich.edu>
9*4882a593Smuzhiyun */
10*4882a593Smuzhiyun
11*4882a593Smuzhiyun /*
12*4882a593Smuzhiyun * Copyright (C) 1998 by the FundsXpress, INC.
13*4882a593Smuzhiyun *
14*4882a593Smuzhiyun * All rights reserved.
15*4882a593Smuzhiyun *
16*4882a593Smuzhiyun * Export of this software from the United States of America may require
17*4882a593Smuzhiyun * a specific license from the United States Government. It is the
18*4882a593Smuzhiyun * responsibility of any person or organization contemplating export to
19*4882a593Smuzhiyun * obtain such a license before exporting.
20*4882a593Smuzhiyun *
21*4882a593Smuzhiyun * WITHIN THAT CONSTRAINT, permission to use, copy, modify, and
22*4882a593Smuzhiyun * distribute this software and its documentation for any purpose and
23*4882a593Smuzhiyun * without fee is hereby granted, provided that the above copyright
24*4882a593Smuzhiyun * notice appear in all copies and that both that copyright notice and
25*4882a593Smuzhiyun * this permission notice appear in supporting documentation, and that
26*4882a593Smuzhiyun * the name of FundsXpress. not be used in advertising or publicity pertaining
27*4882a593Smuzhiyun * to distribution of the software without specific, written prior
28*4882a593Smuzhiyun * permission. FundsXpress makes no representations about the suitability of
29*4882a593Smuzhiyun * this software for any purpose. It is provided "as is" without express
30*4882a593Smuzhiyun * or implied warranty.
31*4882a593Smuzhiyun *
32*4882a593Smuzhiyun * THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR
33*4882a593Smuzhiyun * IMPLIED WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED
34*4882a593Smuzhiyun * WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR A PARTICULAR PURPOSE.
35*4882a593Smuzhiyun */
36*4882a593Smuzhiyun
37*4882a593Smuzhiyun #include <crypto/algapi.h>
38*4882a593Smuzhiyun #include <crypto/hash.h>
39*4882a593Smuzhiyun #include <crypto/skcipher.h>
40*4882a593Smuzhiyun #include <linux/err.h>
41*4882a593Smuzhiyun #include <linux/types.h>
42*4882a593Smuzhiyun #include <linux/mm.h>
43*4882a593Smuzhiyun #include <linux/scatterlist.h>
44*4882a593Smuzhiyun #include <linux/highmem.h>
45*4882a593Smuzhiyun #include <linux/pagemap.h>
46*4882a593Smuzhiyun #include <linux/random.h>
47*4882a593Smuzhiyun #include <linux/sunrpc/gss_krb5.h>
48*4882a593Smuzhiyun #include <linux/sunrpc/xdr.h>
49*4882a593Smuzhiyun
50*4882a593Smuzhiyun #if IS_ENABLED(CONFIG_SUNRPC_DEBUG)
51*4882a593Smuzhiyun # define RPCDBG_FACILITY RPCDBG_AUTH
52*4882a593Smuzhiyun #endif
53*4882a593Smuzhiyun
54*4882a593Smuzhiyun u32
krb5_encrypt(struct crypto_sync_skcipher * tfm,void * iv,void * in,void * out,int length)55*4882a593Smuzhiyun krb5_encrypt(
56*4882a593Smuzhiyun struct crypto_sync_skcipher *tfm,
57*4882a593Smuzhiyun void * iv,
58*4882a593Smuzhiyun void * in,
59*4882a593Smuzhiyun void * out,
60*4882a593Smuzhiyun int length)
61*4882a593Smuzhiyun {
62*4882a593Smuzhiyun u32 ret = -EINVAL;
63*4882a593Smuzhiyun struct scatterlist sg[1];
64*4882a593Smuzhiyun u8 local_iv[GSS_KRB5_MAX_BLOCKSIZE] = {0};
65*4882a593Smuzhiyun SYNC_SKCIPHER_REQUEST_ON_STACK(req, tfm);
66*4882a593Smuzhiyun
67*4882a593Smuzhiyun if (length % crypto_sync_skcipher_blocksize(tfm) != 0)
68*4882a593Smuzhiyun goto out;
69*4882a593Smuzhiyun
70*4882a593Smuzhiyun if (crypto_sync_skcipher_ivsize(tfm) > GSS_KRB5_MAX_BLOCKSIZE) {
71*4882a593Smuzhiyun dprintk("RPC: gss_k5encrypt: tfm iv size too large %d\n",
72*4882a593Smuzhiyun crypto_sync_skcipher_ivsize(tfm));
73*4882a593Smuzhiyun goto out;
74*4882a593Smuzhiyun }
75*4882a593Smuzhiyun
76*4882a593Smuzhiyun if (iv)
77*4882a593Smuzhiyun memcpy(local_iv, iv, crypto_sync_skcipher_ivsize(tfm));
78*4882a593Smuzhiyun
79*4882a593Smuzhiyun memcpy(out, in, length);
80*4882a593Smuzhiyun sg_init_one(sg, out, length);
81*4882a593Smuzhiyun
82*4882a593Smuzhiyun skcipher_request_set_sync_tfm(req, tfm);
83*4882a593Smuzhiyun skcipher_request_set_callback(req, 0, NULL, NULL);
84*4882a593Smuzhiyun skcipher_request_set_crypt(req, sg, sg, length, local_iv);
85*4882a593Smuzhiyun
86*4882a593Smuzhiyun ret = crypto_skcipher_encrypt(req);
87*4882a593Smuzhiyun skcipher_request_zero(req);
88*4882a593Smuzhiyun out:
89*4882a593Smuzhiyun dprintk("RPC: krb5_encrypt returns %d\n", ret);
90*4882a593Smuzhiyun return ret;
91*4882a593Smuzhiyun }
92*4882a593Smuzhiyun
93*4882a593Smuzhiyun u32
krb5_decrypt(struct crypto_sync_skcipher * tfm,void * iv,void * in,void * out,int length)94*4882a593Smuzhiyun krb5_decrypt(
95*4882a593Smuzhiyun struct crypto_sync_skcipher *tfm,
96*4882a593Smuzhiyun void * iv,
97*4882a593Smuzhiyun void * in,
98*4882a593Smuzhiyun void * out,
99*4882a593Smuzhiyun int length)
100*4882a593Smuzhiyun {
101*4882a593Smuzhiyun u32 ret = -EINVAL;
102*4882a593Smuzhiyun struct scatterlist sg[1];
103*4882a593Smuzhiyun u8 local_iv[GSS_KRB5_MAX_BLOCKSIZE] = {0};
104*4882a593Smuzhiyun SYNC_SKCIPHER_REQUEST_ON_STACK(req, tfm);
105*4882a593Smuzhiyun
106*4882a593Smuzhiyun if (length % crypto_sync_skcipher_blocksize(tfm) != 0)
107*4882a593Smuzhiyun goto out;
108*4882a593Smuzhiyun
109*4882a593Smuzhiyun if (crypto_sync_skcipher_ivsize(tfm) > GSS_KRB5_MAX_BLOCKSIZE) {
110*4882a593Smuzhiyun dprintk("RPC: gss_k5decrypt: tfm iv size too large %d\n",
111*4882a593Smuzhiyun crypto_sync_skcipher_ivsize(tfm));
112*4882a593Smuzhiyun goto out;
113*4882a593Smuzhiyun }
114*4882a593Smuzhiyun if (iv)
115*4882a593Smuzhiyun memcpy(local_iv, iv, crypto_sync_skcipher_ivsize(tfm));
116*4882a593Smuzhiyun
117*4882a593Smuzhiyun memcpy(out, in, length);
118*4882a593Smuzhiyun sg_init_one(sg, out, length);
119*4882a593Smuzhiyun
120*4882a593Smuzhiyun skcipher_request_set_sync_tfm(req, tfm);
121*4882a593Smuzhiyun skcipher_request_set_callback(req, 0, NULL, NULL);
122*4882a593Smuzhiyun skcipher_request_set_crypt(req, sg, sg, length, local_iv);
123*4882a593Smuzhiyun
124*4882a593Smuzhiyun ret = crypto_skcipher_decrypt(req);
125*4882a593Smuzhiyun skcipher_request_zero(req);
126*4882a593Smuzhiyun out:
127*4882a593Smuzhiyun dprintk("RPC: gss_k5decrypt returns %d\n",ret);
128*4882a593Smuzhiyun return ret;
129*4882a593Smuzhiyun }
130*4882a593Smuzhiyun
131*4882a593Smuzhiyun static int
checksummer(struct scatterlist * sg,void * data)132*4882a593Smuzhiyun checksummer(struct scatterlist *sg, void *data)
133*4882a593Smuzhiyun {
134*4882a593Smuzhiyun struct ahash_request *req = data;
135*4882a593Smuzhiyun
136*4882a593Smuzhiyun ahash_request_set_crypt(req, sg, NULL, sg->length);
137*4882a593Smuzhiyun
138*4882a593Smuzhiyun return crypto_ahash_update(req);
139*4882a593Smuzhiyun }
140*4882a593Smuzhiyun
141*4882a593Smuzhiyun /*
142*4882a593Smuzhiyun * checksum the plaintext data and hdrlen bytes of the token header
143*4882a593Smuzhiyun * The checksum is performed over the first 8 bytes of the
144*4882a593Smuzhiyun * gss token header and then over the data body
145*4882a593Smuzhiyun */
146*4882a593Smuzhiyun u32
make_checksum(struct krb5_ctx * kctx,char * header,int hdrlen,struct xdr_buf * body,int body_offset,u8 * cksumkey,unsigned int usage,struct xdr_netobj * cksumout)147*4882a593Smuzhiyun make_checksum(struct krb5_ctx *kctx, char *header, int hdrlen,
148*4882a593Smuzhiyun struct xdr_buf *body, int body_offset, u8 *cksumkey,
149*4882a593Smuzhiyun unsigned int usage, struct xdr_netobj *cksumout)
150*4882a593Smuzhiyun {
151*4882a593Smuzhiyun struct crypto_ahash *tfm;
152*4882a593Smuzhiyun struct ahash_request *req;
153*4882a593Smuzhiyun struct scatterlist sg[1];
154*4882a593Smuzhiyun int err = -1;
155*4882a593Smuzhiyun u8 *checksumdata;
156*4882a593Smuzhiyun unsigned int checksumlen;
157*4882a593Smuzhiyun
158*4882a593Smuzhiyun if (cksumout->len < kctx->gk5e->cksumlength) {
159*4882a593Smuzhiyun dprintk("%s: checksum buffer length, %u, too small for %s\n",
160*4882a593Smuzhiyun __func__, cksumout->len, kctx->gk5e->name);
161*4882a593Smuzhiyun return GSS_S_FAILURE;
162*4882a593Smuzhiyun }
163*4882a593Smuzhiyun
164*4882a593Smuzhiyun checksumdata = kmalloc(GSS_KRB5_MAX_CKSUM_LEN, GFP_NOFS);
165*4882a593Smuzhiyun if (checksumdata == NULL)
166*4882a593Smuzhiyun return GSS_S_FAILURE;
167*4882a593Smuzhiyun
168*4882a593Smuzhiyun tfm = crypto_alloc_ahash(kctx->gk5e->cksum_name, 0, CRYPTO_ALG_ASYNC);
169*4882a593Smuzhiyun if (IS_ERR(tfm))
170*4882a593Smuzhiyun goto out_free_cksum;
171*4882a593Smuzhiyun
172*4882a593Smuzhiyun req = ahash_request_alloc(tfm, GFP_NOFS);
173*4882a593Smuzhiyun if (!req)
174*4882a593Smuzhiyun goto out_free_ahash;
175*4882a593Smuzhiyun
176*4882a593Smuzhiyun ahash_request_set_callback(req, CRYPTO_TFM_REQ_MAY_SLEEP, NULL, NULL);
177*4882a593Smuzhiyun
178*4882a593Smuzhiyun checksumlen = crypto_ahash_digestsize(tfm);
179*4882a593Smuzhiyun
180*4882a593Smuzhiyun if (cksumkey != NULL) {
181*4882a593Smuzhiyun err = crypto_ahash_setkey(tfm, cksumkey,
182*4882a593Smuzhiyun kctx->gk5e->keylength);
183*4882a593Smuzhiyun if (err)
184*4882a593Smuzhiyun goto out;
185*4882a593Smuzhiyun }
186*4882a593Smuzhiyun
187*4882a593Smuzhiyun err = crypto_ahash_init(req);
188*4882a593Smuzhiyun if (err)
189*4882a593Smuzhiyun goto out;
190*4882a593Smuzhiyun sg_init_one(sg, header, hdrlen);
191*4882a593Smuzhiyun ahash_request_set_crypt(req, sg, NULL, hdrlen);
192*4882a593Smuzhiyun err = crypto_ahash_update(req);
193*4882a593Smuzhiyun if (err)
194*4882a593Smuzhiyun goto out;
195*4882a593Smuzhiyun err = xdr_process_buf(body, body_offset, body->len - body_offset,
196*4882a593Smuzhiyun checksummer, req);
197*4882a593Smuzhiyun if (err)
198*4882a593Smuzhiyun goto out;
199*4882a593Smuzhiyun ahash_request_set_crypt(req, NULL, checksumdata, 0);
200*4882a593Smuzhiyun err = crypto_ahash_final(req);
201*4882a593Smuzhiyun if (err)
202*4882a593Smuzhiyun goto out;
203*4882a593Smuzhiyun
204*4882a593Smuzhiyun switch (kctx->gk5e->ctype) {
205*4882a593Smuzhiyun case CKSUMTYPE_RSA_MD5:
206*4882a593Smuzhiyun err = kctx->gk5e->encrypt(kctx->seq, NULL, checksumdata,
207*4882a593Smuzhiyun checksumdata, checksumlen);
208*4882a593Smuzhiyun if (err)
209*4882a593Smuzhiyun goto out;
210*4882a593Smuzhiyun memcpy(cksumout->data,
211*4882a593Smuzhiyun checksumdata + checksumlen - kctx->gk5e->cksumlength,
212*4882a593Smuzhiyun kctx->gk5e->cksumlength);
213*4882a593Smuzhiyun break;
214*4882a593Smuzhiyun case CKSUMTYPE_HMAC_SHA1_DES3:
215*4882a593Smuzhiyun memcpy(cksumout->data, checksumdata, kctx->gk5e->cksumlength);
216*4882a593Smuzhiyun break;
217*4882a593Smuzhiyun default:
218*4882a593Smuzhiyun BUG();
219*4882a593Smuzhiyun break;
220*4882a593Smuzhiyun }
221*4882a593Smuzhiyun cksumout->len = kctx->gk5e->cksumlength;
222*4882a593Smuzhiyun out:
223*4882a593Smuzhiyun ahash_request_free(req);
224*4882a593Smuzhiyun out_free_ahash:
225*4882a593Smuzhiyun crypto_free_ahash(tfm);
226*4882a593Smuzhiyun out_free_cksum:
227*4882a593Smuzhiyun kfree(checksumdata);
228*4882a593Smuzhiyun return err ? GSS_S_FAILURE : 0;
229*4882a593Smuzhiyun }
230*4882a593Smuzhiyun
231*4882a593Smuzhiyun /*
232*4882a593Smuzhiyun * checksum the plaintext data and hdrlen bytes of the token header
233*4882a593Smuzhiyun * Per rfc4121, sec. 4.2.4, the checksum is performed over the data
234*4882a593Smuzhiyun * body then over the first 16 octets of the MIC token
235*4882a593Smuzhiyun * Inclusion of the header data in the calculation of the
236*4882a593Smuzhiyun * checksum is optional.
237*4882a593Smuzhiyun */
238*4882a593Smuzhiyun u32
make_checksum_v2(struct krb5_ctx * kctx,char * header,int hdrlen,struct xdr_buf * body,int body_offset,u8 * cksumkey,unsigned int usage,struct xdr_netobj * cksumout)239*4882a593Smuzhiyun make_checksum_v2(struct krb5_ctx *kctx, char *header, int hdrlen,
240*4882a593Smuzhiyun struct xdr_buf *body, int body_offset, u8 *cksumkey,
241*4882a593Smuzhiyun unsigned int usage, struct xdr_netobj *cksumout)
242*4882a593Smuzhiyun {
243*4882a593Smuzhiyun struct crypto_ahash *tfm;
244*4882a593Smuzhiyun struct ahash_request *req;
245*4882a593Smuzhiyun struct scatterlist sg[1];
246*4882a593Smuzhiyun int err = -1;
247*4882a593Smuzhiyun u8 *checksumdata;
248*4882a593Smuzhiyun
249*4882a593Smuzhiyun if (kctx->gk5e->keyed_cksum == 0) {
250*4882a593Smuzhiyun dprintk("%s: expected keyed hash for %s\n",
251*4882a593Smuzhiyun __func__, kctx->gk5e->name);
252*4882a593Smuzhiyun return GSS_S_FAILURE;
253*4882a593Smuzhiyun }
254*4882a593Smuzhiyun if (cksumkey == NULL) {
255*4882a593Smuzhiyun dprintk("%s: no key supplied for %s\n",
256*4882a593Smuzhiyun __func__, kctx->gk5e->name);
257*4882a593Smuzhiyun return GSS_S_FAILURE;
258*4882a593Smuzhiyun }
259*4882a593Smuzhiyun
260*4882a593Smuzhiyun checksumdata = kmalloc(GSS_KRB5_MAX_CKSUM_LEN, GFP_NOFS);
261*4882a593Smuzhiyun if (!checksumdata)
262*4882a593Smuzhiyun return GSS_S_FAILURE;
263*4882a593Smuzhiyun
264*4882a593Smuzhiyun tfm = crypto_alloc_ahash(kctx->gk5e->cksum_name, 0, CRYPTO_ALG_ASYNC);
265*4882a593Smuzhiyun if (IS_ERR(tfm))
266*4882a593Smuzhiyun goto out_free_cksum;
267*4882a593Smuzhiyun
268*4882a593Smuzhiyun req = ahash_request_alloc(tfm, GFP_NOFS);
269*4882a593Smuzhiyun if (!req)
270*4882a593Smuzhiyun goto out_free_ahash;
271*4882a593Smuzhiyun
272*4882a593Smuzhiyun ahash_request_set_callback(req, CRYPTO_TFM_REQ_MAY_SLEEP, NULL, NULL);
273*4882a593Smuzhiyun
274*4882a593Smuzhiyun err = crypto_ahash_setkey(tfm, cksumkey, kctx->gk5e->keylength);
275*4882a593Smuzhiyun if (err)
276*4882a593Smuzhiyun goto out;
277*4882a593Smuzhiyun
278*4882a593Smuzhiyun err = crypto_ahash_init(req);
279*4882a593Smuzhiyun if (err)
280*4882a593Smuzhiyun goto out;
281*4882a593Smuzhiyun err = xdr_process_buf(body, body_offset, body->len - body_offset,
282*4882a593Smuzhiyun checksummer, req);
283*4882a593Smuzhiyun if (err)
284*4882a593Smuzhiyun goto out;
285*4882a593Smuzhiyun if (header != NULL) {
286*4882a593Smuzhiyun sg_init_one(sg, header, hdrlen);
287*4882a593Smuzhiyun ahash_request_set_crypt(req, sg, NULL, hdrlen);
288*4882a593Smuzhiyun err = crypto_ahash_update(req);
289*4882a593Smuzhiyun if (err)
290*4882a593Smuzhiyun goto out;
291*4882a593Smuzhiyun }
292*4882a593Smuzhiyun ahash_request_set_crypt(req, NULL, checksumdata, 0);
293*4882a593Smuzhiyun err = crypto_ahash_final(req);
294*4882a593Smuzhiyun if (err)
295*4882a593Smuzhiyun goto out;
296*4882a593Smuzhiyun
297*4882a593Smuzhiyun cksumout->len = kctx->gk5e->cksumlength;
298*4882a593Smuzhiyun
299*4882a593Smuzhiyun switch (kctx->gk5e->ctype) {
300*4882a593Smuzhiyun case CKSUMTYPE_HMAC_SHA1_96_AES128:
301*4882a593Smuzhiyun case CKSUMTYPE_HMAC_SHA1_96_AES256:
302*4882a593Smuzhiyun /* note that this truncates the hash */
303*4882a593Smuzhiyun memcpy(cksumout->data, checksumdata, kctx->gk5e->cksumlength);
304*4882a593Smuzhiyun break;
305*4882a593Smuzhiyun default:
306*4882a593Smuzhiyun BUG();
307*4882a593Smuzhiyun break;
308*4882a593Smuzhiyun }
309*4882a593Smuzhiyun out:
310*4882a593Smuzhiyun ahash_request_free(req);
311*4882a593Smuzhiyun out_free_ahash:
312*4882a593Smuzhiyun crypto_free_ahash(tfm);
313*4882a593Smuzhiyun out_free_cksum:
314*4882a593Smuzhiyun kfree(checksumdata);
315*4882a593Smuzhiyun return err ? GSS_S_FAILURE : 0;
316*4882a593Smuzhiyun }
317*4882a593Smuzhiyun
318*4882a593Smuzhiyun struct encryptor_desc {
319*4882a593Smuzhiyun u8 iv[GSS_KRB5_MAX_BLOCKSIZE];
320*4882a593Smuzhiyun struct skcipher_request *req;
321*4882a593Smuzhiyun int pos;
322*4882a593Smuzhiyun struct xdr_buf *outbuf;
323*4882a593Smuzhiyun struct page **pages;
324*4882a593Smuzhiyun struct scatterlist infrags[4];
325*4882a593Smuzhiyun struct scatterlist outfrags[4];
326*4882a593Smuzhiyun int fragno;
327*4882a593Smuzhiyun int fraglen;
328*4882a593Smuzhiyun };
329*4882a593Smuzhiyun
330*4882a593Smuzhiyun static int
encryptor(struct scatterlist * sg,void * data)331*4882a593Smuzhiyun encryptor(struct scatterlist *sg, void *data)
332*4882a593Smuzhiyun {
333*4882a593Smuzhiyun struct encryptor_desc *desc = data;
334*4882a593Smuzhiyun struct xdr_buf *outbuf = desc->outbuf;
335*4882a593Smuzhiyun struct crypto_sync_skcipher *tfm =
336*4882a593Smuzhiyun crypto_sync_skcipher_reqtfm(desc->req);
337*4882a593Smuzhiyun struct page *in_page;
338*4882a593Smuzhiyun int thislen = desc->fraglen + sg->length;
339*4882a593Smuzhiyun int fraglen, ret;
340*4882a593Smuzhiyun int page_pos;
341*4882a593Smuzhiyun
342*4882a593Smuzhiyun /* Worst case is 4 fragments: head, end of page 1, start
343*4882a593Smuzhiyun * of page 2, tail. Anything more is a bug. */
344*4882a593Smuzhiyun BUG_ON(desc->fragno > 3);
345*4882a593Smuzhiyun
346*4882a593Smuzhiyun page_pos = desc->pos - outbuf->head[0].iov_len;
347*4882a593Smuzhiyun if (page_pos >= 0 && page_pos < outbuf->page_len) {
348*4882a593Smuzhiyun /* pages are not in place: */
349*4882a593Smuzhiyun int i = (page_pos + outbuf->page_base) >> PAGE_SHIFT;
350*4882a593Smuzhiyun in_page = desc->pages[i];
351*4882a593Smuzhiyun } else {
352*4882a593Smuzhiyun in_page = sg_page(sg);
353*4882a593Smuzhiyun }
354*4882a593Smuzhiyun sg_set_page(&desc->infrags[desc->fragno], in_page, sg->length,
355*4882a593Smuzhiyun sg->offset);
356*4882a593Smuzhiyun sg_set_page(&desc->outfrags[desc->fragno], sg_page(sg), sg->length,
357*4882a593Smuzhiyun sg->offset);
358*4882a593Smuzhiyun desc->fragno++;
359*4882a593Smuzhiyun desc->fraglen += sg->length;
360*4882a593Smuzhiyun desc->pos += sg->length;
361*4882a593Smuzhiyun
362*4882a593Smuzhiyun fraglen = thislen & (crypto_sync_skcipher_blocksize(tfm) - 1);
363*4882a593Smuzhiyun thislen -= fraglen;
364*4882a593Smuzhiyun
365*4882a593Smuzhiyun if (thislen == 0)
366*4882a593Smuzhiyun return 0;
367*4882a593Smuzhiyun
368*4882a593Smuzhiyun sg_mark_end(&desc->infrags[desc->fragno - 1]);
369*4882a593Smuzhiyun sg_mark_end(&desc->outfrags[desc->fragno - 1]);
370*4882a593Smuzhiyun
371*4882a593Smuzhiyun skcipher_request_set_crypt(desc->req, desc->infrags, desc->outfrags,
372*4882a593Smuzhiyun thislen, desc->iv);
373*4882a593Smuzhiyun
374*4882a593Smuzhiyun ret = crypto_skcipher_encrypt(desc->req);
375*4882a593Smuzhiyun if (ret)
376*4882a593Smuzhiyun return ret;
377*4882a593Smuzhiyun
378*4882a593Smuzhiyun sg_init_table(desc->infrags, 4);
379*4882a593Smuzhiyun sg_init_table(desc->outfrags, 4);
380*4882a593Smuzhiyun
381*4882a593Smuzhiyun if (fraglen) {
382*4882a593Smuzhiyun sg_set_page(&desc->outfrags[0], sg_page(sg), fraglen,
383*4882a593Smuzhiyun sg->offset + sg->length - fraglen);
384*4882a593Smuzhiyun desc->infrags[0] = desc->outfrags[0];
385*4882a593Smuzhiyun sg_assign_page(&desc->infrags[0], in_page);
386*4882a593Smuzhiyun desc->fragno = 1;
387*4882a593Smuzhiyun desc->fraglen = fraglen;
388*4882a593Smuzhiyun } else {
389*4882a593Smuzhiyun desc->fragno = 0;
390*4882a593Smuzhiyun desc->fraglen = 0;
391*4882a593Smuzhiyun }
392*4882a593Smuzhiyun return 0;
393*4882a593Smuzhiyun }
394*4882a593Smuzhiyun
395*4882a593Smuzhiyun int
gss_encrypt_xdr_buf(struct crypto_sync_skcipher * tfm,struct xdr_buf * buf,int offset,struct page ** pages)396*4882a593Smuzhiyun gss_encrypt_xdr_buf(struct crypto_sync_skcipher *tfm, struct xdr_buf *buf,
397*4882a593Smuzhiyun int offset, struct page **pages)
398*4882a593Smuzhiyun {
399*4882a593Smuzhiyun int ret;
400*4882a593Smuzhiyun struct encryptor_desc desc;
401*4882a593Smuzhiyun SYNC_SKCIPHER_REQUEST_ON_STACK(req, tfm);
402*4882a593Smuzhiyun
403*4882a593Smuzhiyun BUG_ON((buf->len - offset) % crypto_sync_skcipher_blocksize(tfm) != 0);
404*4882a593Smuzhiyun
405*4882a593Smuzhiyun skcipher_request_set_sync_tfm(req, tfm);
406*4882a593Smuzhiyun skcipher_request_set_callback(req, 0, NULL, NULL);
407*4882a593Smuzhiyun
408*4882a593Smuzhiyun memset(desc.iv, 0, sizeof(desc.iv));
409*4882a593Smuzhiyun desc.req = req;
410*4882a593Smuzhiyun desc.pos = offset;
411*4882a593Smuzhiyun desc.outbuf = buf;
412*4882a593Smuzhiyun desc.pages = pages;
413*4882a593Smuzhiyun desc.fragno = 0;
414*4882a593Smuzhiyun desc.fraglen = 0;
415*4882a593Smuzhiyun
416*4882a593Smuzhiyun sg_init_table(desc.infrags, 4);
417*4882a593Smuzhiyun sg_init_table(desc.outfrags, 4);
418*4882a593Smuzhiyun
419*4882a593Smuzhiyun ret = xdr_process_buf(buf, offset, buf->len - offset, encryptor, &desc);
420*4882a593Smuzhiyun skcipher_request_zero(req);
421*4882a593Smuzhiyun return ret;
422*4882a593Smuzhiyun }
423*4882a593Smuzhiyun
424*4882a593Smuzhiyun struct decryptor_desc {
425*4882a593Smuzhiyun u8 iv[GSS_KRB5_MAX_BLOCKSIZE];
426*4882a593Smuzhiyun struct skcipher_request *req;
427*4882a593Smuzhiyun struct scatterlist frags[4];
428*4882a593Smuzhiyun int fragno;
429*4882a593Smuzhiyun int fraglen;
430*4882a593Smuzhiyun };
431*4882a593Smuzhiyun
432*4882a593Smuzhiyun static int
decryptor(struct scatterlist * sg,void * data)433*4882a593Smuzhiyun decryptor(struct scatterlist *sg, void *data)
434*4882a593Smuzhiyun {
435*4882a593Smuzhiyun struct decryptor_desc *desc = data;
436*4882a593Smuzhiyun int thislen = desc->fraglen + sg->length;
437*4882a593Smuzhiyun struct crypto_sync_skcipher *tfm =
438*4882a593Smuzhiyun crypto_sync_skcipher_reqtfm(desc->req);
439*4882a593Smuzhiyun int fraglen, ret;
440*4882a593Smuzhiyun
441*4882a593Smuzhiyun /* Worst case is 4 fragments: head, end of page 1, start
442*4882a593Smuzhiyun * of page 2, tail. Anything more is a bug. */
443*4882a593Smuzhiyun BUG_ON(desc->fragno > 3);
444*4882a593Smuzhiyun sg_set_page(&desc->frags[desc->fragno], sg_page(sg), sg->length,
445*4882a593Smuzhiyun sg->offset);
446*4882a593Smuzhiyun desc->fragno++;
447*4882a593Smuzhiyun desc->fraglen += sg->length;
448*4882a593Smuzhiyun
449*4882a593Smuzhiyun fraglen = thislen & (crypto_sync_skcipher_blocksize(tfm) - 1);
450*4882a593Smuzhiyun thislen -= fraglen;
451*4882a593Smuzhiyun
452*4882a593Smuzhiyun if (thislen == 0)
453*4882a593Smuzhiyun return 0;
454*4882a593Smuzhiyun
455*4882a593Smuzhiyun sg_mark_end(&desc->frags[desc->fragno - 1]);
456*4882a593Smuzhiyun
457*4882a593Smuzhiyun skcipher_request_set_crypt(desc->req, desc->frags, desc->frags,
458*4882a593Smuzhiyun thislen, desc->iv);
459*4882a593Smuzhiyun
460*4882a593Smuzhiyun ret = crypto_skcipher_decrypt(desc->req);
461*4882a593Smuzhiyun if (ret)
462*4882a593Smuzhiyun return ret;
463*4882a593Smuzhiyun
464*4882a593Smuzhiyun sg_init_table(desc->frags, 4);
465*4882a593Smuzhiyun
466*4882a593Smuzhiyun if (fraglen) {
467*4882a593Smuzhiyun sg_set_page(&desc->frags[0], sg_page(sg), fraglen,
468*4882a593Smuzhiyun sg->offset + sg->length - fraglen);
469*4882a593Smuzhiyun desc->fragno = 1;
470*4882a593Smuzhiyun desc->fraglen = fraglen;
471*4882a593Smuzhiyun } else {
472*4882a593Smuzhiyun desc->fragno = 0;
473*4882a593Smuzhiyun desc->fraglen = 0;
474*4882a593Smuzhiyun }
475*4882a593Smuzhiyun return 0;
476*4882a593Smuzhiyun }
477*4882a593Smuzhiyun
478*4882a593Smuzhiyun int
gss_decrypt_xdr_buf(struct crypto_sync_skcipher * tfm,struct xdr_buf * buf,int offset)479*4882a593Smuzhiyun gss_decrypt_xdr_buf(struct crypto_sync_skcipher *tfm, struct xdr_buf *buf,
480*4882a593Smuzhiyun int offset)
481*4882a593Smuzhiyun {
482*4882a593Smuzhiyun int ret;
483*4882a593Smuzhiyun struct decryptor_desc desc;
484*4882a593Smuzhiyun SYNC_SKCIPHER_REQUEST_ON_STACK(req, tfm);
485*4882a593Smuzhiyun
486*4882a593Smuzhiyun /* XXXJBF: */
487*4882a593Smuzhiyun BUG_ON((buf->len - offset) % crypto_sync_skcipher_blocksize(tfm) != 0);
488*4882a593Smuzhiyun
489*4882a593Smuzhiyun skcipher_request_set_sync_tfm(req, tfm);
490*4882a593Smuzhiyun skcipher_request_set_callback(req, 0, NULL, NULL);
491*4882a593Smuzhiyun
492*4882a593Smuzhiyun memset(desc.iv, 0, sizeof(desc.iv));
493*4882a593Smuzhiyun desc.req = req;
494*4882a593Smuzhiyun desc.fragno = 0;
495*4882a593Smuzhiyun desc.fraglen = 0;
496*4882a593Smuzhiyun
497*4882a593Smuzhiyun sg_init_table(desc.frags, 4);
498*4882a593Smuzhiyun
499*4882a593Smuzhiyun ret = xdr_process_buf(buf, offset, buf->len - offset, decryptor, &desc);
500*4882a593Smuzhiyun skcipher_request_zero(req);
501*4882a593Smuzhiyun return ret;
502*4882a593Smuzhiyun }
503*4882a593Smuzhiyun
504*4882a593Smuzhiyun /*
505*4882a593Smuzhiyun * This function makes the assumption that it was ultimately called
506*4882a593Smuzhiyun * from gss_wrap().
507*4882a593Smuzhiyun *
508*4882a593Smuzhiyun * The client auth_gss code moves any existing tail data into a
509*4882a593Smuzhiyun * separate page before calling gss_wrap.
510*4882a593Smuzhiyun * The server svcauth_gss code ensures that both the head and the
511*4882a593Smuzhiyun * tail have slack space of RPC_MAX_AUTH_SIZE before calling gss_wrap.
512*4882a593Smuzhiyun *
513*4882a593Smuzhiyun * Even with that guarantee, this function may be called more than
514*4882a593Smuzhiyun * once in the processing of gss_wrap(). The best we can do is
515*4882a593Smuzhiyun * verify at compile-time (see GSS_KRB5_SLACK_CHECK) that the
516*4882a593Smuzhiyun * largest expected shift will fit within RPC_MAX_AUTH_SIZE.
517*4882a593Smuzhiyun * At run-time we can verify that a single invocation of this
518*4882a593Smuzhiyun * function doesn't attempt to use more the RPC_MAX_AUTH_SIZE.
519*4882a593Smuzhiyun */
520*4882a593Smuzhiyun
521*4882a593Smuzhiyun int
xdr_extend_head(struct xdr_buf * buf,unsigned int base,unsigned int shiftlen)522*4882a593Smuzhiyun xdr_extend_head(struct xdr_buf *buf, unsigned int base, unsigned int shiftlen)
523*4882a593Smuzhiyun {
524*4882a593Smuzhiyun u8 *p;
525*4882a593Smuzhiyun
526*4882a593Smuzhiyun if (shiftlen == 0)
527*4882a593Smuzhiyun return 0;
528*4882a593Smuzhiyun
529*4882a593Smuzhiyun BUILD_BUG_ON(GSS_KRB5_MAX_SLACK_NEEDED > RPC_MAX_AUTH_SIZE);
530*4882a593Smuzhiyun BUG_ON(shiftlen > RPC_MAX_AUTH_SIZE);
531*4882a593Smuzhiyun
532*4882a593Smuzhiyun p = buf->head[0].iov_base + base;
533*4882a593Smuzhiyun
534*4882a593Smuzhiyun memmove(p + shiftlen, p, buf->head[0].iov_len - base);
535*4882a593Smuzhiyun
536*4882a593Smuzhiyun buf->head[0].iov_len += shiftlen;
537*4882a593Smuzhiyun buf->len += shiftlen;
538*4882a593Smuzhiyun
539*4882a593Smuzhiyun return 0;
540*4882a593Smuzhiyun }
541*4882a593Smuzhiyun
542*4882a593Smuzhiyun static u32
gss_krb5_cts_crypt(struct crypto_sync_skcipher * cipher,struct xdr_buf * buf,u32 offset,u8 * iv,struct page ** pages,int encrypt)543*4882a593Smuzhiyun gss_krb5_cts_crypt(struct crypto_sync_skcipher *cipher, struct xdr_buf *buf,
544*4882a593Smuzhiyun u32 offset, u8 *iv, struct page **pages, int encrypt)
545*4882a593Smuzhiyun {
546*4882a593Smuzhiyun u32 ret;
547*4882a593Smuzhiyun struct scatterlist sg[1];
548*4882a593Smuzhiyun SYNC_SKCIPHER_REQUEST_ON_STACK(req, cipher);
549*4882a593Smuzhiyun u8 *data;
550*4882a593Smuzhiyun struct page **save_pages;
551*4882a593Smuzhiyun u32 len = buf->len - offset;
552*4882a593Smuzhiyun
553*4882a593Smuzhiyun if (len > GSS_KRB5_MAX_BLOCKSIZE * 2) {
554*4882a593Smuzhiyun WARN_ON(0);
555*4882a593Smuzhiyun return -ENOMEM;
556*4882a593Smuzhiyun }
557*4882a593Smuzhiyun data = kmalloc(GSS_KRB5_MAX_BLOCKSIZE * 2, GFP_NOFS);
558*4882a593Smuzhiyun if (!data)
559*4882a593Smuzhiyun return -ENOMEM;
560*4882a593Smuzhiyun
561*4882a593Smuzhiyun /*
562*4882a593Smuzhiyun * For encryption, we want to read from the cleartext
563*4882a593Smuzhiyun * page cache pages, and write the encrypted data to
564*4882a593Smuzhiyun * the supplied xdr_buf pages.
565*4882a593Smuzhiyun */
566*4882a593Smuzhiyun save_pages = buf->pages;
567*4882a593Smuzhiyun if (encrypt)
568*4882a593Smuzhiyun buf->pages = pages;
569*4882a593Smuzhiyun
570*4882a593Smuzhiyun ret = read_bytes_from_xdr_buf(buf, offset, data, len);
571*4882a593Smuzhiyun buf->pages = save_pages;
572*4882a593Smuzhiyun if (ret)
573*4882a593Smuzhiyun goto out;
574*4882a593Smuzhiyun
575*4882a593Smuzhiyun sg_init_one(sg, data, len);
576*4882a593Smuzhiyun
577*4882a593Smuzhiyun skcipher_request_set_sync_tfm(req, cipher);
578*4882a593Smuzhiyun skcipher_request_set_callback(req, 0, NULL, NULL);
579*4882a593Smuzhiyun skcipher_request_set_crypt(req, sg, sg, len, iv);
580*4882a593Smuzhiyun
581*4882a593Smuzhiyun if (encrypt)
582*4882a593Smuzhiyun ret = crypto_skcipher_encrypt(req);
583*4882a593Smuzhiyun else
584*4882a593Smuzhiyun ret = crypto_skcipher_decrypt(req);
585*4882a593Smuzhiyun
586*4882a593Smuzhiyun skcipher_request_zero(req);
587*4882a593Smuzhiyun
588*4882a593Smuzhiyun if (ret)
589*4882a593Smuzhiyun goto out;
590*4882a593Smuzhiyun
591*4882a593Smuzhiyun ret = write_bytes_to_xdr_buf(buf, offset, data, len);
592*4882a593Smuzhiyun
593*4882a593Smuzhiyun out:
594*4882a593Smuzhiyun kfree(data);
595*4882a593Smuzhiyun return ret;
596*4882a593Smuzhiyun }
597*4882a593Smuzhiyun
598*4882a593Smuzhiyun u32
gss_krb5_aes_encrypt(struct krb5_ctx * kctx,u32 offset,struct xdr_buf * buf,struct page ** pages)599*4882a593Smuzhiyun gss_krb5_aes_encrypt(struct krb5_ctx *kctx, u32 offset,
600*4882a593Smuzhiyun struct xdr_buf *buf, struct page **pages)
601*4882a593Smuzhiyun {
602*4882a593Smuzhiyun u32 err;
603*4882a593Smuzhiyun struct xdr_netobj hmac;
604*4882a593Smuzhiyun u8 *cksumkey;
605*4882a593Smuzhiyun u8 *ecptr;
606*4882a593Smuzhiyun struct crypto_sync_skcipher *cipher, *aux_cipher;
607*4882a593Smuzhiyun int blocksize;
608*4882a593Smuzhiyun struct page **save_pages;
609*4882a593Smuzhiyun int nblocks, nbytes;
610*4882a593Smuzhiyun struct encryptor_desc desc;
611*4882a593Smuzhiyun u32 cbcbytes;
612*4882a593Smuzhiyun unsigned int usage;
613*4882a593Smuzhiyun
614*4882a593Smuzhiyun if (kctx->initiate) {
615*4882a593Smuzhiyun cipher = kctx->initiator_enc;
616*4882a593Smuzhiyun aux_cipher = kctx->initiator_enc_aux;
617*4882a593Smuzhiyun cksumkey = kctx->initiator_integ;
618*4882a593Smuzhiyun usage = KG_USAGE_INITIATOR_SEAL;
619*4882a593Smuzhiyun } else {
620*4882a593Smuzhiyun cipher = kctx->acceptor_enc;
621*4882a593Smuzhiyun aux_cipher = kctx->acceptor_enc_aux;
622*4882a593Smuzhiyun cksumkey = kctx->acceptor_integ;
623*4882a593Smuzhiyun usage = KG_USAGE_ACCEPTOR_SEAL;
624*4882a593Smuzhiyun }
625*4882a593Smuzhiyun blocksize = crypto_sync_skcipher_blocksize(cipher);
626*4882a593Smuzhiyun
627*4882a593Smuzhiyun /* hide the gss token header and insert the confounder */
628*4882a593Smuzhiyun offset += GSS_KRB5_TOK_HDR_LEN;
629*4882a593Smuzhiyun if (xdr_extend_head(buf, offset, kctx->gk5e->conflen))
630*4882a593Smuzhiyun return GSS_S_FAILURE;
631*4882a593Smuzhiyun gss_krb5_make_confounder(buf->head[0].iov_base + offset, kctx->gk5e->conflen);
632*4882a593Smuzhiyun offset -= GSS_KRB5_TOK_HDR_LEN;
633*4882a593Smuzhiyun
634*4882a593Smuzhiyun if (buf->tail[0].iov_base != NULL) {
635*4882a593Smuzhiyun ecptr = buf->tail[0].iov_base + buf->tail[0].iov_len;
636*4882a593Smuzhiyun } else {
637*4882a593Smuzhiyun buf->tail[0].iov_base = buf->head[0].iov_base
638*4882a593Smuzhiyun + buf->head[0].iov_len;
639*4882a593Smuzhiyun buf->tail[0].iov_len = 0;
640*4882a593Smuzhiyun ecptr = buf->tail[0].iov_base;
641*4882a593Smuzhiyun }
642*4882a593Smuzhiyun
643*4882a593Smuzhiyun /* copy plaintext gss token header after filler (if any) */
644*4882a593Smuzhiyun memcpy(ecptr, buf->head[0].iov_base + offset, GSS_KRB5_TOK_HDR_LEN);
645*4882a593Smuzhiyun buf->tail[0].iov_len += GSS_KRB5_TOK_HDR_LEN;
646*4882a593Smuzhiyun buf->len += GSS_KRB5_TOK_HDR_LEN;
647*4882a593Smuzhiyun
648*4882a593Smuzhiyun /* Do the HMAC */
649*4882a593Smuzhiyun hmac.len = GSS_KRB5_MAX_CKSUM_LEN;
650*4882a593Smuzhiyun hmac.data = buf->tail[0].iov_base + buf->tail[0].iov_len;
651*4882a593Smuzhiyun
652*4882a593Smuzhiyun /*
653*4882a593Smuzhiyun * When we are called, pages points to the real page cache
654*4882a593Smuzhiyun * data -- which we can't go and encrypt! buf->pages points
655*4882a593Smuzhiyun * to scratch pages which we are going to send off to the
656*4882a593Smuzhiyun * client/server. Swap in the plaintext pages to calculate
657*4882a593Smuzhiyun * the hmac.
658*4882a593Smuzhiyun */
659*4882a593Smuzhiyun save_pages = buf->pages;
660*4882a593Smuzhiyun buf->pages = pages;
661*4882a593Smuzhiyun
662*4882a593Smuzhiyun err = make_checksum_v2(kctx, NULL, 0, buf,
663*4882a593Smuzhiyun offset + GSS_KRB5_TOK_HDR_LEN,
664*4882a593Smuzhiyun cksumkey, usage, &hmac);
665*4882a593Smuzhiyun buf->pages = save_pages;
666*4882a593Smuzhiyun if (err)
667*4882a593Smuzhiyun return GSS_S_FAILURE;
668*4882a593Smuzhiyun
669*4882a593Smuzhiyun nbytes = buf->len - offset - GSS_KRB5_TOK_HDR_LEN;
670*4882a593Smuzhiyun nblocks = (nbytes + blocksize - 1) / blocksize;
671*4882a593Smuzhiyun cbcbytes = 0;
672*4882a593Smuzhiyun if (nblocks > 2)
673*4882a593Smuzhiyun cbcbytes = (nblocks - 2) * blocksize;
674*4882a593Smuzhiyun
675*4882a593Smuzhiyun memset(desc.iv, 0, sizeof(desc.iv));
676*4882a593Smuzhiyun
677*4882a593Smuzhiyun if (cbcbytes) {
678*4882a593Smuzhiyun SYNC_SKCIPHER_REQUEST_ON_STACK(req, aux_cipher);
679*4882a593Smuzhiyun
680*4882a593Smuzhiyun desc.pos = offset + GSS_KRB5_TOK_HDR_LEN;
681*4882a593Smuzhiyun desc.fragno = 0;
682*4882a593Smuzhiyun desc.fraglen = 0;
683*4882a593Smuzhiyun desc.pages = pages;
684*4882a593Smuzhiyun desc.outbuf = buf;
685*4882a593Smuzhiyun desc.req = req;
686*4882a593Smuzhiyun
687*4882a593Smuzhiyun skcipher_request_set_sync_tfm(req, aux_cipher);
688*4882a593Smuzhiyun skcipher_request_set_callback(req, 0, NULL, NULL);
689*4882a593Smuzhiyun
690*4882a593Smuzhiyun sg_init_table(desc.infrags, 4);
691*4882a593Smuzhiyun sg_init_table(desc.outfrags, 4);
692*4882a593Smuzhiyun
693*4882a593Smuzhiyun err = xdr_process_buf(buf, offset + GSS_KRB5_TOK_HDR_LEN,
694*4882a593Smuzhiyun cbcbytes, encryptor, &desc);
695*4882a593Smuzhiyun skcipher_request_zero(req);
696*4882a593Smuzhiyun if (err)
697*4882a593Smuzhiyun goto out_err;
698*4882a593Smuzhiyun }
699*4882a593Smuzhiyun
700*4882a593Smuzhiyun /* Make sure IV carries forward from any CBC results. */
701*4882a593Smuzhiyun err = gss_krb5_cts_crypt(cipher, buf,
702*4882a593Smuzhiyun offset + GSS_KRB5_TOK_HDR_LEN + cbcbytes,
703*4882a593Smuzhiyun desc.iv, pages, 1);
704*4882a593Smuzhiyun if (err) {
705*4882a593Smuzhiyun err = GSS_S_FAILURE;
706*4882a593Smuzhiyun goto out_err;
707*4882a593Smuzhiyun }
708*4882a593Smuzhiyun
709*4882a593Smuzhiyun /* Now update buf to account for HMAC */
710*4882a593Smuzhiyun buf->tail[0].iov_len += kctx->gk5e->cksumlength;
711*4882a593Smuzhiyun buf->len += kctx->gk5e->cksumlength;
712*4882a593Smuzhiyun
713*4882a593Smuzhiyun out_err:
714*4882a593Smuzhiyun if (err)
715*4882a593Smuzhiyun err = GSS_S_FAILURE;
716*4882a593Smuzhiyun return err;
717*4882a593Smuzhiyun }
718*4882a593Smuzhiyun
719*4882a593Smuzhiyun u32
gss_krb5_aes_decrypt(struct krb5_ctx * kctx,u32 offset,u32 len,struct xdr_buf * buf,u32 * headskip,u32 * tailskip)720*4882a593Smuzhiyun gss_krb5_aes_decrypt(struct krb5_ctx *kctx, u32 offset, u32 len,
721*4882a593Smuzhiyun struct xdr_buf *buf, u32 *headskip, u32 *tailskip)
722*4882a593Smuzhiyun {
723*4882a593Smuzhiyun struct xdr_buf subbuf;
724*4882a593Smuzhiyun u32 ret = 0;
725*4882a593Smuzhiyun u8 *cksum_key;
726*4882a593Smuzhiyun struct crypto_sync_skcipher *cipher, *aux_cipher;
727*4882a593Smuzhiyun struct xdr_netobj our_hmac_obj;
728*4882a593Smuzhiyun u8 our_hmac[GSS_KRB5_MAX_CKSUM_LEN];
729*4882a593Smuzhiyun u8 pkt_hmac[GSS_KRB5_MAX_CKSUM_LEN];
730*4882a593Smuzhiyun int nblocks, blocksize, cbcbytes;
731*4882a593Smuzhiyun struct decryptor_desc desc;
732*4882a593Smuzhiyun unsigned int usage;
733*4882a593Smuzhiyun
734*4882a593Smuzhiyun if (kctx->initiate) {
735*4882a593Smuzhiyun cipher = kctx->acceptor_enc;
736*4882a593Smuzhiyun aux_cipher = kctx->acceptor_enc_aux;
737*4882a593Smuzhiyun cksum_key = kctx->acceptor_integ;
738*4882a593Smuzhiyun usage = KG_USAGE_ACCEPTOR_SEAL;
739*4882a593Smuzhiyun } else {
740*4882a593Smuzhiyun cipher = kctx->initiator_enc;
741*4882a593Smuzhiyun aux_cipher = kctx->initiator_enc_aux;
742*4882a593Smuzhiyun cksum_key = kctx->initiator_integ;
743*4882a593Smuzhiyun usage = KG_USAGE_INITIATOR_SEAL;
744*4882a593Smuzhiyun }
745*4882a593Smuzhiyun blocksize = crypto_sync_skcipher_blocksize(cipher);
746*4882a593Smuzhiyun
747*4882a593Smuzhiyun
748*4882a593Smuzhiyun /* create a segment skipping the header and leaving out the checksum */
749*4882a593Smuzhiyun xdr_buf_subsegment(buf, &subbuf, offset + GSS_KRB5_TOK_HDR_LEN,
750*4882a593Smuzhiyun (len - offset - GSS_KRB5_TOK_HDR_LEN -
751*4882a593Smuzhiyun kctx->gk5e->cksumlength));
752*4882a593Smuzhiyun
753*4882a593Smuzhiyun nblocks = (subbuf.len + blocksize - 1) / blocksize;
754*4882a593Smuzhiyun
755*4882a593Smuzhiyun cbcbytes = 0;
756*4882a593Smuzhiyun if (nblocks > 2)
757*4882a593Smuzhiyun cbcbytes = (nblocks - 2) * blocksize;
758*4882a593Smuzhiyun
759*4882a593Smuzhiyun memset(desc.iv, 0, sizeof(desc.iv));
760*4882a593Smuzhiyun
761*4882a593Smuzhiyun if (cbcbytes) {
762*4882a593Smuzhiyun SYNC_SKCIPHER_REQUEST_ON_STACK(req, aux_cipher);
763*4882a593Smuzhiyun
764*4882a593Smuzhiyun desc.fragno = 0;
765*4882a593Smuzhiyun desc.fraglen = 0;
766*4882a593Smuzhiyun desc.req = req;
767*4882a593Smuzhiyun
768*4882a593Smuzhiyun skcipher_request_set_sync_tfm(req, aux_cipher);
769*4882a593Smuzhiyun skcipher_request_set_callback(req, 0, NULL, NULL);
770*4882a593Smuzhiyun
771*4882a593Smuzhiyun sg_init_table(desc.frags, 4);
772*4882a593Smuzhiyun
773*4882a593Smuzhiyun ret = xdr_process_buf(&subbuf, 0, cbcbytes, decryptor, &desc);
774*4882a593Smuzhiyun skcipher_request_zero(req);
775*4882a593Smuzhiyun if (ret)
776*4882a593Smuzhiyun goto out_err;
777*4882a593Smuzhiyun }
778*4882a593Smuzhiyun
779*4882a593Smuzhiyun /* Make sure IV carries forward from any CBC results. */
780*4882a593Smuzhiyun ret = gss_krb5_cts_crypt(cipher, &subbuf, cbcbytes, desc.iv, NULL, 0);
781*4882a593Smuzhiyun if (ret)
782*4882a593Smuzhiyun goto out_err;
783*4882a593Smuzhiyun
784*4882a593Smuzhiyun
785*4882a593Smuzhiyun /* Calculate our hmac over the plaintext data */
786*4882a593Smuzhiyun our_hmac_obj.len = sizeof(our_hmac);
787*4882a593Smuzhiyun our_hmac_obj.data = our_hmac;
788*4882a593Smuzhiyun
789*4882a593Smuzhiyun ret = make_checksum_v2(kctx, NULL, 0, &subbuf, 0,
790*4882a593Smuzhiyun cksum_key, usage, &our_hmac_obj);
791*4882a593Smuzhiyun if (ret)
792*4882a593Smuzhiyun goto out_err;
793*4882a593Smuzhiyun
794*4882a593Smuzhiyun /* Get the packet's hmac value */
795*4882a593Smuzhiyun ret = read_bytes_from_xdr_buf(buf, len - kctx->gk5e->cksumlength,
796*4882a593Smuzhiyun pkt_hmac, kctx->gk5e->cksumlength);
797*4882a593Smuzhiyun if (ret)
798*4882a593Smuzhiyun goto out_err;
799*4882a593Smuzhiyun
800*4882a593Smuzhiyun if (crypto_memneq(pkt_hmac, our_hmac, kctx->gk5e->cksumlength) != 0) {
801*4882a593Smuzhiyun ret = GSS_S_BAD_SIG;
802*4882a593Smuzhiyun goto out_err;
803*4882a593Smuzhiyun }
804*4882a593Smuzhiyun *headskip = kctx->gk5e->conflen;
805*4882a593Smuzhiyun *tailskip = kctx->gk5e->cksumlength;
806*4882a593Smuzhiyun out_err:
807*4882a593Smuzhiyun if (ret && ret != GSS_S_BAD_SIG)
808*4882a593Smuzhiyun ret = GSS_S_FAILURE;
809*4882a593Smuzhiyun return ret;
810*4882a593Smuzhiyun }
811