des_s390.c 6.6 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3
/*
 * Cryptographic API.
 *
4
 * s390 implementation of the DES Cipher Algorithm.
L
Linus Torvalds 已提交
5 6 7 8 9 10 11 12 13 14 15 16 17 18
 *
 * Copyright (c) 2003 IBM Deutschland Entwicklung GmbH, IBM Corporation
 * Author(s): Thomas Spatzier (tspat@de.ibm.com)
 *
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 of the License, or
 * (at your option) any later version.
 *
 */
#include <linux/init.h>
#include <linux/module.h>
#include <linux/crypto.h>
19

20
#include "crypt_s390.h"
L
Linus Torvalds 已提交
21 22 23 24 25 26 27 28 29 30 31
#include "crypto_des.h"

#define DES_BLOCK_SIZE 8
#define DES_KEY_SIZE 8

#define DES3_128_KEY_SIZE	(2 * DES_KEY_SIZE)
#define DES3_128_BLOCK_SIZE	DES_BLOCK_SIZE

#define DES3_192_KEY_SIZE	(3 * DES_KEY_SIZE)
#define DES3_192_BLOCK_SIZE	DES_BLOCK_SIZE

32
struct crypt_s390_des_ctx {
L
Linus Torvalds 已提交
33 34 35 36
	u8 iv[DES_BLOCK_SIZE];
	u8 key[DES_KEY_SIZE];
};

37
struct crypt_s390_des3_128_ctx {
L
Linus Torvalds 已提交
38 39 40 41
	u8 iv[DES_BLOCK_SIZE];
	u8 key[DES3_128_KEY_SIZE];
};

42
struct crypt_s390_des3_192_ctx {
L
Linus Torvalds 已提交
43 44 45 46
	u8 iv[DES_BLOCK_SIZE];
	u8 key[DES3_192_KEY_SIZE];
};

47 48
static int des_setkey(void *ctx, const u8 *key, unsigned int keylen,
		      u32 *flags)
L
Linus Torvalds 已提交
49
{
50
	struct crypt_s390_des_ctx *dctx = ctx;
L
Linus Torvalds 已提交
51 52
	int ret;

53
	/* test if key is valid (not a weak key) */
L
Linus Torvalds 已提交
54
	ret = crypto_des_check_key(key, keylen, flags);
55
	if (ret == 0)
L
Linus Torvalds 已提交
56 57 58 59
		memcpy(dctx->key, key, keylen);
	return ret;
}

60
static void des_encrypt(void *ctx, u8 *dst, const u8 *src)
L
Linus Torvalds 已提交
61
{
62
	struct crypt_s390_des_ctx *dctx = ctx;
L
Linus Torvalds 已提交
63

64
	crypt_s390_km(KM_DEA_ENCRYPT, dctx->key, dst, src, DES_BLOCK_SIZE);
L
Linus Torvalds 已提交
65 66
}

67
static void des_decrypt(void *ctx, u8 *dst, const u8 *src)
L
Linus Torvalds 已提交
68
{
69
	struct crypt_s390_des_ctx *dctx = ctx;
L
Linus Torvalds 已提交
70

71
	crypt_s390_km(KM_DEA_DECRYPT, dctx->key, dst, src, DES_BLOCK_SIZE);
L
Linus Torvalds 已提交
72 73 74 75 76 77
}

static struct crypto_alg des_alg = {
	.cra_name		=	"des",
	.cra_flags		=	CRYPTO_ALG_TYPE_CIPHER,
	.cra_blocksize		=	DES_BLOCK_SIZE,
78
	.cra_ctxsize		=	sizeof(struct crypt_s390_des_ctx),
L
Linus Torvalds 已提交
79 80
	.cra_module		=	THIS_MODULE,
	.cra_list		=	LIST_HEAD_INIT(des_alg.cra_list),
81 82 83 84 85 86 87 88 89
	.cra_u			=	{
		.cipher = {
			.cia_min_keysize	=	DES_KEY_SIZE,
			.cia_max_keysize	=	DES_KEY_SIZE,
			.cia_setkey		=	des_setkey,
			.cia_encrypt		=	des_encrypt,
			.cia_decrypt		=	des_decrypt
		}
	}
L
Linus Torvalds 已提交
90 91 92 93 94 95 96 97 98 99 100 101 102 103
};

/*
 * RFC2451:
 *
 *   For DES-EDE3, there is no known need to reject weak or
 *   complementation keys.  Any weakness is obviated by the use of
 *   multiple keys.
 *
 *   However, if the two  independent 64-bit keys are equal,
 *   then the DES3 operation is simply the same as DES.
 *   Implementers MUST reject keys that exhibit this property.
 *
 */
104 105
static int des3_128_setkey(void *ctx, const u8 *key, unsigned int keylen,
			   u32 *flags)
L
Linus Torvalds 已提交
106 107
{
	int i, ret;
108
	struct crypt_s390_des3_128_ctx *dctx = ctx;
L
Linus Torvalds 已提交
109 110 111 112 113 114
	const u8* temp_key = key;

	if (!(memcmp(key, &key[DES_KEY_SIZE], DES_KEY_SIZE))) {
		*flags |= CRYPTO_TFM_RES_BAD_KEY_SCHED;
		return -EINVAL;
	}
115
	for (i = 0; i < 2; i++, temp_key += DES_KEY_SIZE) {
L
Linus Torvalds 已提交
116 117 118 119 120 121 122 123
		ret = crypto_des_check_key(temp_key, DES_KEY_SIZE, flags);
		if (ret < 0)
			return ret;
	}
	memcpy(dctx->key, key, keylen);
	return 0;
}

124
static void des3_128_encrypt(void *ctx, u8 *dst, const u8 *src)
L
Linus Torvalds 已提交
125
{
126
	struct crypt_s390_des3_128_ctx *dctx = ctx;
L
Linus Torvalds 已提交
127

128
	crypt_s390_km(KM_TDEA_128_ENCRYPT, dctx->key, dst, (void*)src,
129
		      DES3_128_BLOCK_SIZE);
L
Linus Torvalds 已提交
130 131
}

132
static void des3_128_decrypt(void *ctx, u8 *dst, const u8 *src)
L
Linus Torvalds 已提交
133
{
134
	struct crypt_s390_des3_128_ctx *dctx = ctx;
L
Linus Torvalds 已提交
135

136
	crypt_s390_km(KM_TDEA_128_DECRYPT, dctx->key, dst, (void*)src,
137
		      DES3_128_BLOCK_SIZE);
L
Linus Torvalds 已提交
138 139 140 141 142 143
}

static struct crypto_alg des3_128_alg = {
	.cra_name		=	"des3_ede128",
	.cra_flags		=	CRYPTO_ALG_TYPE_CIPHER,
	.cra_blocksize		=	DES3_128_BLOCK_SIZE,
144
	.cra_ctxsize		=	sizeof(struct crypt_s390_des3_128_ctx),
L
Linus Torvalds 已提交
145 146
	.cra_module		=	THIS_MODULE,
	.cra_list		=	LIST_HEAD_INIT(des3_128_alg.cra_list),
147 148 149 150 151 152 153 154 155
	.cra_u			=	{
		.cipher = {
			.cia_min_keysize	=	DES3_128_KEY_SIZE,
			.cia_max_keysize	=	DES3_128_KEY_SIZE,
			.cia_setkey		=	des3_128_setkey,
			.cia_encrypt		=	des3_128_encrypt,
			.cia_decrypt		=	des3_128_decrypt
		}
	}
L
Linus Torvalds 已提交
156 157 158 159 160 161 162 163 164 165 166 167 168 169 170
};

/*
 * RFC2451:
 *
 *   For DES-EDE3, there is no known need to reject weak or
 *   complementation keys.  Any weakness is obviated by the use of
 *   multiple keys.
 *
 *   However, if the first two or last two independent 64-bit keys are
 *   equal (k1 == k2 or k2 == k3), then the DES3 operation is simply the
 *   same as DES.  Implementers MUST reject keys that exhibit this
 *   property.
 *
 */
171 172
static int des3_192_setkey(void *ctx, const u8 *key, unsigned int keylen,
			   u32 *flags)
L
Linus Torvalds 已提交
173 174
{
	int i, ret;
175 176
	struct crypt_s390_des3_192_ctx *dctx = ctx;
	const u8* temp_key = key;
L
Linus Torvalds 已提交
177 178 179

	if (!(memcmp(key, &key[DES_KEY_SIZE], DES_KEY_SIZE) &&
	    memcmp(&key[DES_KEY_SIZE], &key[DES_KEY_SIZE * 2],
180
		   DES_KEY_SIZE))) {
L
Linus Torvalds 已提交
181 182 183 184 185 186

		*flags |= CRYPTO_TFM_RES_BAD_KEY_SCHED;
		return -EINVAL;
	}
	for (i = 0; i < 3; i++, temp_key += DES_KEY_SIZE) {
		ret = crypto_des_check_key(temp_key, DES_KEY_SIZE, flags);
187
		if (ret < 0)
L
Linus Torvalds 已提交
188 189 190 191 192 193
			return ret;
	}
	memcpy(dctx->key, key, keylen);
	return 0;
}

194
static void des3_192_encrypt(void *ctx, u8 *dst, const u8 *src)
L
Linus Torvalds 已提交
195
{
196
	struct crypt_s390_des3_192_ctx *dctx = ctx;
L
Linus Torvalds 已提交
197

198
	crypt_s390_km(KM_TDEA_192_ENCRYPT, dctx->key, dst, (void*)src,
199
		      DES3_192_BLOCK_SIZE);
L
Linus Torvalds 已提交
200 201
}

202
static void des3_192_decrypt(void *ctx, u8 *dst, const u8 *src)
L
Linus Torvalds 已提交
203
{
204
	struct crypt_s390_des3_192_ctx *dctx = ctx;
L
Linus Torvalds 已提交
205

206
	crypt_s390_km(KM_TDEA_192_DECRYPT, dctx->key, dst, (void*)src,
207
		      DES3_192_BLOCK_SIZE);
L
Linus Torvalds 已提交
208 209 210 211 212 213
}

static struct crypto_alg des3_192_alg = {
	.cra_name		=	"des3_ede",
	.cra_flags		=	CRYPTO_ALG_TYPE_CIPHER,
	.cra_blocksize		=	DES3_192_BLOCK_SIZE,
214
	.cra_ctxsize		=	sizeof(struct crypt_s390_des3_192_ctx),
L
Linus Torvalds 已提交
215 216
	.cra_module		=	THIS_MODULE,
	.cra_list		=	LIST_HEAD_INIT(des3_192_alg.cra_list),
217 218 219 220 221 222 223 224 225
	.cra_u			=	{
		.cipher = {
			.cia_min_keysize	=	DES3_192_KEY_SIZE,
			.cia_max_keysize	=	DES3_192_KEY_SIZE,
			.cia_setkey		=	des3_192_setkey,
			.cia_encrypt		=	des3_192_encrypt,
			.cia_decrypt		=	des3_192_decrypt
		}
	}
L
Linus Torvalds 已提交
226 227
};

228
static int init(void)
L
Linus Torvalds 已提交
229
{
230
	int ret = 0;
L
Linus Torvalds 已提交
231

232 233
	if (!crypt_s390_func_available(KM_DEA_ENCRYPT) ||
	    !crypt_s390_func_available(KM_TDEA_128_ENCRYPT) ||
234
	    !crypt_s390_func_available(KM_TDEA_192_ENCRYPT))
L
Linus Torvalds 已提交
235 236
		return -ENOSYS;

237 238 239 240
	ret |= (crypto_register_alg(&des_alg) == 0) ? 0:1;
	ret |= (crypto_register_alg(&des3_128_alg) == 0) ? 0:2;
	ret |= (crypto_register_alg(&des3_192_alg) == 0) ? 0:4;
	if (ret) {
L
Linus Torvalds 已提交
241 242 243 244 245 246 247 248
		crypto_unregister_alg(&des3_192_alg);
		crypto_unregister_alg(&des3_128_alg);
		crypto_unregister_alg(&des_alg);
		return -EEXIST;
	}
	return 0;
}

249
static void __exit fini(void)
L
Linus Torvalds 已提交
250 251 252 253 254 255 256 257 258 259 260 261 262 263
{
	crypto_unregister_alg(&des3_192_alg);
	crypto_unregister_alg(&des3_128_alg);
	crypto_unregister_alg(&des_alg);
}

module_init(init);
module_exit(fini);

MODULE_ALIAS("des");
MODULE_ALIAS("des3_ede");

MODULE_LICENSE("GPL");
MODULE_DESCRIPTION("DES & Triple DES EDE Cipher Algorithms");