xref: /OK3568_Linux_fs/kernel/net/sunrpc/auth_gss/gss_krb5_crypto.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
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