// SPDX-License-Identifier: GPL-2.0-or-later
/*
* Cryptographic API.
*
* ARIA Cipher Algorithm.
*
* Documentation of ARIA can be found in RFC 5794.
* Copyright (c) 2022 Taehee Yoo <ap420073@gmail.com>
*
* Information for ARIA
* http://210.104.33.10/ARIA/index-e.html (English)
* http://seed.kisa.or.kr/ (Korean)
*
* Public domain version is distributed above.
*/
#include <crypto/aria.h>
#include <linux/unaligned.h>
static const u32 key_rc[
20] = {
0x517cc1b7,
0x27220a94,
0xfe13abe8,
0xfa9a6ee0,
0x6db14acc,
0x9e21c820,
0xff28b1d5,
0xef5de2b0,
0xdb92371d,
0x2126e970,
0x03249775,
0x04e8c90e,
0x517cc1b7,
0x27220a94,
0xfe13abe8,
0xfa9a6ee0,
0x6db14acc,
0x9e21c820,
0xff28b1d5,
0xef5de2b0
};
static void aria_set_encrypt_key(
struct aria_ctx *ctx,
const u8 *in_key,
unsigned int key_len)
{
u32 w0[
4], w1[
4], w2[
4], w3[
4];
u32 reg0, reg1, reg2, reg3;
const u32 *ck;
int rkidx =
0;
ck = &key_rc[(key_len -
16) /
2];
w0[
0] = get_unaligned_be32(&in_key[
0]);
w0[
1] = get_unaligned_be32(&in_key[
4]);
w0[
2] = get_unaligned_be32(&in_key[
8]);
w0[
3] = get_unaligned_be32(&in_key[
12]);
reg0 = w0[
0] ^ ck[
0];
reg1 = w0[
1] ^ ck[
1];
reg2 = w0[
2] ^ ck[
2];
reg3 = w0[
3] ^ ck[
3];
aria_subst_diff_odd(®0, ®1, ®2, ®3);
if (key_len >
16) {
w1[
0] = get_unaligned_be32(&in_key[
16]);
w1[
1] = get_unaligned_be32(&in_key[
20]);
if (key_len >
24) {
w1[
2] = get_unaligned_be32(&in_key[
24]);
w1[
3] = get_unaligned_be32(&in_key[
28]);
}
else {
w1[
2] =
0;
w1[
3] =
0;
}
}
else {
w1[
0] =
0;
w1[
1] =
0;
w1[
2] =
0;
w1[
3] =
0;
}
w1[
0] ^= reg0;
w1[
1] ^= reg1;
w1[
2] ^= reg2;
w1[
3] ^= reg3;
reg0 = w1[
0];
reg1 = w1[
1];
reg2 = w1[
2];
reg3 = w1[
3];
reg0 ^= ck[
4];
reg1 ^= ck[
5];
reg2 ^= ck[
6];
reg3 ^= ck[
7];
aria_subst_diff_even(®0, ®1, ®2, ®3);
reg0 ^= w0[
0];
reg1 ^= w0[
1];
reg2 ^= w0[
2];
reg3 ^= w0[
3];
w2[
0] = reg0;
w2[
1] = reg1;
w2[
2] = reg2;
w2[
3] = reg3;
reg0 ^= ck[
8];
reg1 ^= ck[
9];
reg2 ^= ck[
10];
reg3 ^= ck[
11];
aria_subst_diff_odd(®0, ®1, ®2, ®3);
w3[
0] = reg0 ^ w1[
0];
w3[
1] = reg1 ^ w1[
1];
w3[
2] = reg2 ^ w1[
2];
w3[
3] = reg3 ^ w1[
3];
aria_gsrk(ctx->enc_key[rkidx], w0, w1,
19);
rkidx++;
aria_gsrk(ctx->enc_key[rkidx], w1, w2,
19);
rkidx++;
aria_gsrk(ctx->enc_key[rkidx], w2, w3,
19);
rkidx++;
aria_gsrk(ctx->enc_key[rkidx], w3, w0,
19);
rkidx++;
aria_gsrk(ctx->enc_key[rkidx], w0, w1,
31);
rkidx++;
aria_gsrk(ctx->enc_key[rkidx], w1, w2,
31);
rkidx++;
aria_gsrk(ctx->enc_key[rkidx], w2, w3,
31);
rkidx++;
aria_gsrk(ctx->enc_key[rkidx], w3, w0,
31);
rkidx++;
aria_gsrk(ctx->enc_key[rkidx], w0, w1,
67);
rkidx++;
aria_gsrk(ctx->enc_key[rkidx], w1, w2,
67);
rkidx++;
aria_gsrk(ctx->enc_key[rkidx], w2, w3,
67);
rkidx++;
aria_gsrk(ctx->enc_key[rkidx], w3, w0,
67);
rkidx++;
aria_gsrk(ctx->enc_key[rkidx], w0, w1,
97);
if (key_len >
16) {
rkidx++;
aria_gsrk(ctx->enc_key[rkidx], w1, w2,
97);
rkidx++;
aria_gsrk(ctx->enc_key[rkidx], w2, w3,
97);
if (key_len >
24) {
rkidx++;
aria_gsrk(ctx->enc_key[rkidx], w3, w0,
97);
rkidx++;
aria_gsrk(ctx->enc_key[rkidx], w0, w1,
109);
}
}
}
static void aria_set_decrypt_key(
struct aria_ctx *ctx)
{
int i;
for (i =
0; i <
4; i++) {
ctx->dec_key[
0][i] = ctx->enc_key[ctx->rounds][i];
ctx->dec_key[ctx->rounds][i] = ctx->enc_key[
0][i];
}
for (i =
1; i < ctx->rounds; i++) {
ctx->dec_key[i][
0] = aria_m(ctx->enc_key[ctx->rounds - i][
0]);
ctx->dec_key[i][
1] = aria_m(ctx->enc_key[ctx->rounds - i][
1]);
ctx->dec_key[i][
2] = aria_m(ctx->enc_key[ctx->rounds - i][
2]);
ctx->dec_key[i][
3] = aria_m(ctx->enc_key[ctx->rounds - i][
3]);
aria_diff_word(&ctx->dec_key[i][
0], &ctx->dec_key[i][
1],
&ctx->dec_key[i][
2], &ctx->dec_key[i][
3]);
aria_diff_byte(&ctx->dec_key[i][
1],
&ctx->dec_key[i][
2], &ctx->dec_key[i][
3]);
aria_diff_word(&ctx->dec_key[i][
0], &ctx->dec_key[i][
1],
&ctx->dec_key[i][
2], &ctx->dec_key[i][
3]);
}
}
int aria_set_key(
struct crypto_tfm *tfm,
const u8 *in_key,
unsigned int key_len)
{
struct aria_ctx *ctx = crypto_tfm_ctx(tfm);
if (key_len !=
16 && key_len !=
24 && key_len !=
32)
return -EINVAL;
BUILD_BUG_ON(
sizeof(ctx->enc_key) !=
272);
BUILD_BUG_ON(
sizeof(ctx->dec_key) !=
272);
BUILD_BUG_ON(
sizeof(
int) !=
sizeof(ctx->rounds));
ctx->key_length = key_len;
ctx->rounds = (key_len +
32) /
4;
aria_set_encrypt_key(ctx, in_key, key_len);
aria_set_decrypt_key(ctx);
return 0;
}
EXPORT_SYMBOL_GPL(aria_set_key);
static void __aria_crypt(
struct aria_ctx *ctx, u8 *out,
const u8 *in,
u32 key[][ARIA_RD_KEY_WORDS])
{
u32 reg0, reg1, reg2, reg3;
int rounds, rkidx =
0;
rounds = ctx->rounds;
reg0 = get_unaligned_be32(&in[
0]);
reg1 = get_unaligned_be32(&in[
4]);
reg2 = get_unaligned_be32(&in[
8]);
reg3 = get_unaligned_be32(&in[
12]);
aria_add_round_key(key[rkidx], ®0, ®1, ®2, ®3);
rkidx++;
aria_subst_diff_odd(®0, ®1, ®2, ®3);
aria_add_round_key(key[rkidx], ®0, ®1, ®2, ®3);
rkidx++;
while ((rounds -=
2) >
0) {
aria_subst_diff_even(®0, ®1, ®2, ®3);
aria_add_round_key(key[rkidx], ®0, ®1, ®2, ®3);
rkidx++;
aria_subst_diff_odd(®0, ®1, ®2, ®3);
aria_add_round_key(key[rkidx], ®0, ®1, ®2, ®3);
rkidx++;
}
reg0 = key[rkidx][
0] ^ make_u32((u8)(x1[get_u8(reg0,
0)]),
(u8)(x2[get_u8(reg0,
1)] >>
8),
(u8)(s1[get_u8(reg0,
2)]),
(u8)(s2[get_u8(reg0,
3)]));
reg1 = key[rkidx][
1] ^ make_u32((u8)(x1[get_u8(reg1,
0)]),
(u8)(x2[get_u8(reg1,
1)] >>
8),
(u8)(s1[get_u8(reg1,
2)]),
(u8)(s2[get_u8(reg1,
3)]));
reg2 = key[rkidx][
2] ^ make_u32((u8)(x1[get_u8(reg2,
0)]),
(u8)(x2[get_u8(reg2,
1)] >>
8),
(u8)(s1[get_u8(reg2,
2)]),
(u8)(s2[get_u8(reg2,
3)]));
reg3 = key[rkidx][
3] ^ make_u32((u8)(x1[get_u8(reg3,
0)]),
(u8)(x2[get_u8(reg3,
1)] >>
8),
(u8)(s1[get_u8(reg3,
2)]),
(u8)(s2[get_u8(reg3,
3)]));
put_unaligned_be32(reg0, &out[
0]);
put_unaligned_be32(reg1, &out[
4]);
put_unaligned_be32(reg2, &out[
8]);
put_unaligned_be32(reg3, &out[
12]);
}
void aria_encrypt(
void *_ctx, u8 *out,
const u8 *in)
{
struct aria_ctx *ctx = (
struct aria_ctx *)_ctx;
__aria_crypt(ctx, out, in, ctx->enc_key);
}
EXPORT_SYMBOL_GPL(aria_encrypt);
void aria_decrypt(
void *_ctx, u8 *out,
const u8 *in)
{
struct aria_ctx *ctx = (
struct aria_ctx *)_ctx;
__aria_crypt(ctx, out, in, ctx->dec_key);
}
EXPORT_SYMBOL_GPL(aria_decrypt);
static void __aria_encrypt(
struct crypto_tfm *tfm, u8 *out,
const u8 *in)
{
struct aria_ctx *ctx = crypto_tfm_ctx(tfm);
__aria_crypt(ctx, out, in, ctx->enc_key);
}
static void __aria_decrypt(
struct crypto_tfm *tfm, u8 *out,
const u8 *in)
{
struct aria_ctx *ctx = crypto_tfm_ctx(tfm);
__aria_crypt(ctx, out, in, ctx->dec_key);
}
static struct crypto_alg aria_alg = {
.cra_name =
"aria",
.cra_driver_name =
"aria-generic",
.cra_priority =
100,
.cra_flags = CRYPTO_ALG_TYPE_CIPHER,
.cra_blocksize = ARIA_BLOCK_SIZE,
.cra_ctxsize =
sizeof(
struct aria_ctx),
.cra_module = THIS_MODULE,
.cra_u = {
.cipher = {
.cia_min_keysize = ARIA_MIN_KEY_SIZE,
.cia_max_keysize = ARIA_MAX_KEY_SIZE,
.cia_setkey = aria_set_key,
.cia_encrypt = __aria_encrypt,
.cia_decrypt = __aria_decrypt
}
}
};
static int __init aria_init(
void)
{
return crypto_register_alg(&aria_alg);
}
static void __
exit aria_fini(
void)
{
crypto_unregister_alg(&aria_alg);
}
module_init(aria_init);
module_exit(aria_fini);
MODULE_DESCRIPTION(
"ARIA Cipher Algorithm");
MODULE_LICENSE(
"GPL");
MODULE_AUTHOR(
"Taehee Yoo <ap420073@gmail.com>");
MODULE_ALIAS_CRYPTO(
"aria");
MODULE_ALIAS_CRYPTO(
"aria-generic");