xref: /freebsd/sys/geom/eli/g_eli_crypto.c (revision 31d62a73c2e6ac0ff413a7a17700ffc7dce254ef)
1 /*-
2  * SPDX-License-Identifier: BSD-2-Clause-FreeBSD
3  *
4  * Copyright (c) 2005-2010 Pawel Jakub Dawidek <pjd@FreeBSD.org>
5  * All rights reserved.
6  *
7  * Redistribution and use in source and binary forms, with or without
8  * modification, are permitted provided that the following conditions
9  * are met:
10  * 1. Redistributions of source code must retain the above copyright
11  *    notice, this list of conditions and the following disclaimer.
12  * 2. Redistributions in binary form must reproduce the above copyright
13  *    notice, this list of conditions and the following disclaimer in the
14  *    documentation and/or other materials provided with the distribution.
15  *
16  * THIS SOFTWARE IS PROVIDED BY THE AUTHORS AND CONTRIBUTORS ``AS IS'' AND
17  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
19  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHORS OR CONTRIBUTORS BE LIABLE
20  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
21  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
22  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
23  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
24  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
25  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
26  * SUCH DAMAGE.
27  */
28 
29 #include <sys/cdefs.h>
30 __FBSDID("$FreeBSD$");
31 
32 #include <sys/param.h>
33 #ifdef _KERNEL
34 #include <sys/systm.h>
35 #include <sys/kernel.h>
36 #include <sys/malloc.h>
37 #else
38 #include <stdint.h>
39 #include <string.h>
40 #include <strings.h>
41 #include <errno.h>
42 #include <assert.h>
43 #include <openssl/evp.h>
44 #define	_OpenSSL_
45 #endif
46 #include <geom/eli/g_eli.h>
47 
48 #ifdef _KERNEL
49 MALLOC_DECLARE(M_ELI);
50 
51 static int
52 g_eli_crypto_done(struct cryptop *crp)
53 {
54 
55 	crp->crp_opaque = (void *)crp;
56 	wakeup(crp);
57 	return (0);
58 }
59 
60 static int
61 g_eli_crypto_cipher(u_int algo, int enc, u_char *data, size_t datasize,
62     const u_char *key, size_t keysize)
63 {
64 	struct cryptoini cri;
65 	struct cryptop *crp;
66 	struct cryptodesc *crd;
67 	crypto_session_t sid;
68 	u_char *p;
69 	int error;
70 
71 	KASSERT(algo != CRYPTO_AES_XTS,
72 	    ("%s: CRYPTO_AES_XTS unexpected here", __func__));
73 
74 	bzero(&cri, sizeof(cri));
75 	cri.cri_alg = algo;
76 	cri.cri_key = __DECONST(void *, key);
77 	cri.cri_klen = keysize;
78 	error = crypto_newsession(&sid, &cri, CRYPTOCAP_F_SOFTWARE);
79 	if (error != 0)
80 		return (error);
81 	p = malloc(sizeof(*crp) + sizeof(*crd), M_ELI, M_NOWAIT | M_ZERO);
82 	if (p == NULL) {
83 		crypto_freesession(sid);
84 		return (ENOMEM);
85 	}
86 	crp = (struct cryptop *)p;	p += sizeof(*crp);
87 	crd = (struct cryptodesc *)p;	p += sizeof(*crd);
88 
89 	crd->crd_skip = 0;
90 	crd->crd_len = datasize;
91 	crd->crd_flags = CRD_F_IV_EXPLICIT | CRD_F_IV_PRESENT;
92 	if (enc)
93 		crd->crd_flags |= CRD_F_ENCRYPT;
94 	crd->crd_alg = algo;
95 	crd->crd_key = __DECONST(void *, key);
96 	crd->crd_klen = keysize;
97 	bzero(crd->crd_iv, sizeof(crd->crd_iv));
98 	crd->crd_next = NULL;
99 
100 	crp->crp_session = sid;
101 	crp->crp_ilen = datasize;
102 	crp->crp_olen = datasize;
103 	crp->crp_opaque = NULL;
104 	crp->crp_callback = g_eli_crypto_done;
105 	crp->crp_buf = (void *)data;
106 	crp->crp_flags = CRYPTO_F_CBIFSYNC;
107 	crp->crp_desc = crd;
108 
109 	error = crypto_dispatch(crp);
110 	if (error == 0) {
111 		while (crp->crp_opaque == NULL)
112 			tsleep(crp, PRIBIO, "geli", hz / 5);
113 		error = crp->crp_etype;
114 	}
115 
116 	free(crp, M_ELI);
117 	crypto_freesession(sid);
118 	return (error);
119 }
120 #else	/* !_KERNEL */
121 static int
122 g_eli_crypto_cipher(u_int algo, int enc, u_char *data, size_t datasize,
123     const u_char *key, size_t keysize)
124 {
125 	EVP_CIPHER_CTX *ctx;
126 	const EVP_CIPHER *type;
127 	u_char iv[keysize];
128 	int outsize;
129 
130 	assert(algo != CRYPTO_AES_XTS);
131 
132 	switch (algo) {
133 	case CRYPTO_NULL_CBC:
134 		type = EVP_enc_null();
135 		break;
136 	case CRYPTO_AES_CBC:
137 		switch (keysize) {
138 		case 128:
139 			type = EVP_aes_128_cbc();
140 			break;
141 		case 192:
142 			type = EVP_aes_192_cbc();
143 			break;
144 		case 256:
145 			type = EVP_aes_256_cbc();
146 			break;
147 		default:
148 			return (EINVAL);
149 		}
150 		break;
151 	case CRYPTO_BLF_CBC:
152 		type = EVP_bf_cbc();
153 		break;
154 #ifndef OPENSSL_NO_CAMELLIA
155 	case CRYPTO_CAMELLIA_CBC:
156 		switch (keysize) {
157 		case 128:
158 			type = EVP_camellia_128_cbc();
159 			break;
160 		case 192:
161 			type = EVP_camellia_192_cbc();
162 			break;
163 		case 256:
164 			type = EVP_camellia_256_cbc();
165 			break;
166 		default:
167 			return (EINVAL);
168 		}
169 		break;
170 #endif
171 	case CRYPTO_3DES_CBC:
172 		type = EVP_des_ede3_cbc();
173 		break;
174 	default:
175 		return (EINVAL);
176 	}
177 
178 	ctx = EVP_CIPHER_CTX_new();
179 	if (ctx == NULL)
180 		return (ENOMEM);
181 
182 	EVP_CipherInit_ex(ctx, type, NULL, NULL, NULL, enc);
183 	EVP_CIPHER_CTX_set_key_length(ctx, keysize / 8);
184 	EVP_CIPHER_CTX_set_padding(ctx, 0);
185 	bzero(iv, sizeof(iv));
186 	EVP_CipherInit_ex(ctx, NULL, NULL, key, iv, enc);
187 
188 	if (EVP_CipherUpdate(ctx, data, &outsize, data, datasize) == 0) {
189 		EVP_CIPHER_CTX_free(ctx);
190 		return (EINVAL);
191 	}
192 	assert(outsize == (int)datasize);
193 
194 	if (EVP_CipherFinal_ex(ctx, data + outsize, &outsize) == 0) {
195 		EVP_CIPHER_CTX_free(ctx);
196 		return (EINVAL);
197 	}
198 	assert(outsize == 0);
199 
200 	EVP_CIPHER_CTX_free(ctx);
201 	return (0);
202 }
203 #endif	/* !_KERNEL */
204 
205 int
206 g_eli_crypto_encrypt(u_int algo, u_char *data, size_t datasize,
207     const u_char *key, size_t keysize)
208 {
209 
210 	/* We prefer AES-CBC for metadata protection. */
211 	if (algo == CRYPTO_AES_XTS)
212 		algo = CRYPTO_AES_CBC;
213 
214 	return (g_eli_crypto_cipher(algo, 1, data, datasize, key, keysize));
215 }
216 
217 int
218 g_eli_crypto_decrypt(u_int algo, u_char *data, size_t datasize,
219     const u_char *key, size_t keysize)
220 {
221 
222 	/* We prefer AES-CBC for metadata protection. */
223 	if (algo == CRYPTO_AES_XTS)
224 		algo = CRYPTO_AES_CBC;
225 
226 	return (g_eli_crypto_cipher(algo, 0, data, datasize, key, keysize));
227 }
228