2
0
mirror of git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git synced 2025-09-04 20:19:47 +08:00

crypto: cts - Convert to skcipher

This patch converts cts over to the skcipher interface.  It also
optimises the implementation to use one CBC operation for all but
the last block, which is then processed separately.

Signed-off-by: Herbert Xu <herbert@gondor.apana.org.au>
This commit is contained in:
Herbert Xu 2016-07-12 13:17:48 +08:00
parent 499a66e6b6
commit 0605c41cc5

View File

@ -40,7 +40,7 @@
* rfc3962 includes errata information in its Appendix A. * rfc3962 includes errata information in its Appendix A.
*/ */
#include <crypto/algapi.h> #include <crypto/internal/skcipher.h>
#include <linux/err.h> #include <linux/err.h>
#include <linux/init.h> #include <linux/init.h>
#include <linux/kernel.h> #include <linux/kernel.h>
@ -51,289 +51,364 @@
#include <linux/slab.h> #include <linux/slab.h>
struct crypto_cts_ctx { struct crypto_cts_ctx {
struct crypto_blkcipher *child; struct crypto_skcipher *child;
}; };
static int crypto_cts_setkey(struct crypto_tfm *parent, const u8 *key, struct crypto_cts_reqctx {
struct scatterlist sg[2];
unsigned offset;
struct skcipher_request subreq;
};
static inline u8 *crypto_cts_reqctx_space(struct skcipher_request *req)
{
struct crypto_cts_reqctx *rctx = skcipher_request_ctx(req);
struct crypto_skcipher *tfm = crypto_skcipher_reqtfm(req);
struct crypto_cts_ctx *ctx = crypto_skcipher_ctx(tfm);
struct crypto_skcipher *child = ctx->child;
return PTR_ALIGN((u8 *)(rctx + 1) + crypto_skcipher_reqsize(child),
crypto_skcipher_alignmask(tfm) + 1);
}
static int crypto_cts_setkey(struct crypto_skcipher *parent, const u8 *key,
unsigned int keylen) unsigned int keylen)
{ {
struct crypto_cts_ctx *ctx = crypto_tfm_ctx(parent); struct crypto_cts_ctx *ctx = crypto_skcipher_ctx(parent);
struct crypto_blkcipher *child = ctx->child; struct crypto_skcipher *child = ctx->child;
int err; int err;
crypto_blkcipher_clear_flags(child, CRYPTO_TFM_REQ_MASK); crypto_skcipher_clear_flags(child, CRYPTO_TFM_REQ_MASK);
crypto_blkcipher_set_flags(child, crypto_tfm_get_flags(parent) & crypto_skcipher_set_flags(child, crypto_skcipher_get_flags(parent) &
CRYPTO_TFM_REQ_MASK); CRYPTO_TFM_REQ_MASK);
err = crypto_blkcipher_setkey(child, key, keylen); err = crypto_skcipher_setkey(child, key, keylen);
crypto_tfm_set_flags(parent, crypto_blkcipher_get_flags(child) & crypto_skcipher_set_flags(parent, crypto_skcipher_get_flags(child) &
CRYPTO_TFM_RES_MASK); CRYPTO_TFM_RES_MASK);
return err; return err;
} }
static int cts_cbc_encrypt(struct crypto_cts_ctx *ctx, static void cts_cbc_crypt_done(struct crypto_async_request *areq, int err)
struct blkcipher_desc *desc,
struct scatterlist *dst,
struct scatterlist *src,
unsigned int offset,
unsigned int nbytes)
{ {
int bsize = crypto_blkcipher_blocksize(desc->tfm); struct skcipher_request *req = areq->data;
u8 tmp[bsize], tmp2[bsize];
struct blkcipher_desc lcldesc;
struct scatterlist sgsrc[1], sgdst[1];
int lastn = nbytes - bsize;
u8 iv[bsize];
u8 s[bsize * 2], d[bsize * 2];
int err;
if (lastn < 0) if (err == -EINPROGRESS)
return -EINVAL; return;
sg_init_table(sgsrc, 1); skcipher_request_complete(req, err);
sg_init_table(sgdst, 1);
memset(s, 0, sizeof(s));
scatterwalk_map_and_copy(s, src, offset, nbytes, 0);
memcpy(iv, desc->info, bsize);
lcldesc.tfm = ctx->child;
lcldesc.info = iv;
lcldesc.flags = desc->flags;
sg_set_buf(&sgsrc[0], s, bsize);
sg_set_buf(&sgdst[0], tmp, bsize);
err = crypto_blkcipher_encrypt_iv(&lcldesc, sgdst, sgsrc, bsize);
memcpy(d + bsize, tmp, lastn);
lcldesc.info = tmp;
sg_set_buf(&sgsrc[0], s + bsize, bsize);
sg_set_buf(&sgdst[0], tmp2, bsize);
err = crypto_blkcipher_encrypt_iv(&lcldesc, sgdst, sgsrc, bsize);
memcpy(d, tmp2, bsize);
scatterwalk_map_and_copy(d, dst, offset, nbytes, 1);
memcpy(desc->info, tmp2, bsize);
return err;
} }
static int crypto_cts_encrypt(struct blkcipher_desc *desc, static int cts_cbc_encrypt(struct skcipher_request *req)
struct scatterlist *dst, struct scatterlist *src,
unsigned int nbytes)
{ {
struct crypto_cts_ctx *ctx = crypto_blkcipher_ctx(desc->tfm); struct crypto_cts_reqctx *rctx = skcipher_request_ctx(req);
int bsize = crypto_blkcipher_blocksize(desc->tfm); struct crypto_skcipher *tfm = crypto_skcipher_reqtfm(req);
int tot_blocks = (nbytes + bsize - 1) / bsize; struct skcipher_request *subreq = &rctx->subreq;
int cbc_blocks = tot_blocks > 2 ? tot_blocks - 2 : 0; int bsize = crypto_skcipher_blocksize(tfm);
struct blkcipher_desc lcldesc; u8 d[bsize * 2] __attribute__ ((aligned(__alignof__(u32))));
int err; struct scatterlist *sg;
unsigned int offset;
int lastn;
lcldesc.tfm = ctx->child; offset = rctx->offset;
lcldesc.info = desc->info; lastn = req->cryptlen - offset;
lcldesc.flags = desc->flags;
if (tot_blocks == 1) { sg = scatterwalk_ffwd(rctx->sg, req->dst, offset - bsize);
err = crypto_blkcipher_encrypt_iv(&lcldesc, dst, src, bsize); scatterwalk_map_and_copy(d + bsize, sg, 0, bsize, 0);
} else if (nbytes <= bsize * 2) {
err = cts_cbc_encrypt(ctx, desc, dst, src, 0, nbytes); memset(d, 0, bsize);
} else { scatterwalk_map_and_copy(d, req->src, offset, lastn, 0);
/* do normal function for tot_blocks - 2 */
err = crypto_blkcipher_encrypt_iv(&lcldesc, dst, src, scatterwalk_map_and_copy(d, sg, 0, bsize + lastn, 1);
cbc_blocks * bsize); memzero_explicit(d, sizeof(d));
if (err == 0) {
/* do cts for final two blocks */ skcipher_request_set_callback(subreq, req->base.flags &
err = cts_cbc_encrypt(ctx, desc, dst, src, CRYPTO_TFM_REQ_MAY_BACKLOG,
cbc_blocks * bsize, cts_cbc_crypt_done, req);
nbytes - (cbc_blocks * bsize)); skcipher_request_set_crypt(subreq, sg, sg, bsize, req->iv);
} return crypto_skcipher_encrypt(subreq);
} }
return err; static void crypto_cts_encrypt_done(struct crypto_async_request *areq, int err)
}
static int cts_cbc_decrypt(struct crypto_cts_ctx *ctx,
struct blkcipher_desc *desc,
struct scatterlist *dst,
struct scatterlist *src,
unsigned int offset,
unsigned int nbytes)
{ {
int bsize = crypto_blkcipher_blocksize(desc->tfm); struct skcipher_request *req = areq->data;
u8 tmp[bsize];
struct blkcipher_desc lcldesc;
struct scatterlist sgsrc[1], sgdst[1];
int lastn = nbytes - bsize;
u8 iv[bsize];
u8 s[bsize * 2], d[bsize * 2];
int err;
if (lastn < 0)
return -EINVAL;
sg_init_table(sgsrc, 1);
sg_init_table(sgdst, 1);
scatterwalk_map_and_copy(s, src, offset, nbytes, 0);
lcldesc.tfm = ctx->child;
lcldesc.info = iv;
lcldesc.flags = desc->flags;
/* 1. Decrypt Cn-1 (s) to create Dn (tmp)*/
memset(iv, 0, sizeof(iv));
sg_set_buf(&sgsrc[0], s, bsize);
sg_set_buf(&sgdst[0], tmp, bsize);
err = crypto_blkcipher_decrypt_iv(&lcldesc, sgdst, sgsrc, bsize);
if (err) if (err)
return err; goto out;
err = cts_cbc_encrypt(req);
if (err == -EINPROGRESS ||
(err == -EBUSY && req->base.flags & CRYPTO_TFM_REQ_MAY_BACKLOG))
return;
out:
skcipher_request_complete(req, err);
}
static int crypto_cts_encrypt(struct skcipher_request *req)
{
struct crypto_skcipher *tfm = crypto_skcipher_reqtfm(req);
struct crypto_cts_reqctx *rctx = skcipher_request_ctx(req);
struct crypto_cts_ctx *ctx = crypto_skcipher_ctx(tfm);
struct skcipher_request *subreq = &rctx->subreq;
int bsize = crypto_skcipher_blocksize(tfm);
unsigned int nbytes = req->cryptlen;
int cbc_blocks = (nbytes + bsize - 1) / bsize - 1;
unsigned int offset;
skcipher_request_set_tfm(subreq, ctx->child);
if (cbc_blocks <= 0) {
skcipher_request_set_callback(subreq, req->base.flags,
req->base.complete,
req->base.data);
skcipher_request_set_crypt(subreq, req->src, req->dst, nbytes,
req->iv);
return crypto_skcipher_encrypt(subreq);
}
offset = cbc_blocks * bsize;
rctx->offset = offset;
skcipher_request_set_callback(subreq, req->base.flags,
crypto_cts_encrypt_done, req);
skcipher_request_set_crypt(subreq, req->src, req->dst,
offset, req->iv);
return crypto_skcipher_encrypt(subreq) ?:
cts_cbc_encrypt(req);
}
static int cts_cbc_decrypt(struct skcipher_request *req)
{
struct crypto_cts_reqctx *rctx = skcipher_request_ctx(req);
struct crypto_skcipher *tfm = crypto_skcipher_reqtfm(req);
struct skcipher_request *subreq = &rctx->subreq;
int bsize = crypto_skcipher_blocksize(tfm);
u8 d[bsize * 2] __attribute__ ((aligned(__alignof__(u32))));
struct scatterlist *sg;
unsigned int offset;
u8 *space;
int lastn;
offset = rctx->offset;
lastn = req->cryptlen - offset;
sg = scatterwalk_ffwd(rctx->sg, req->dst, offset - bsize);
/* 1. Decrypt Cn-1 (s) to create Dn */
scatterwalk_map_and_copy(d + bsize, sg, 0, bsize, 0);
space = crypto_cts_reqctx_space(req);
crypto_xor(d + bsize, space, bsize);
/* 2. Pad Cn with zeros at the end to create C of length BB */ /* 2. Pad Cn with zeros at the end to create C of length BB */
memset(iv, 0, sizeof(iv)); memset(d, 0, bsize);
memcpy(iv, s + bsize, lastn); scatterwalk_map_and_copy(d, req->src, offset, lastn, 0);
/* 3. Exclusive-or Dn (tmp) with C (iv) to create Xn (tmp) */ /* 3. Exclusive-or Dn with C to create Xn */
crypto_xor(tmp, iv, bsize); /* 4. Select the first Ln bytes of Xn to create Pn */
/* 4. Select the first Ln bytes of Xn (tmp) to create Pn */ crypto_xor(d + bsize, d, lastn);
memcpy(d + bsize, tmp, lastn);
/* 5. Append the tail (BB - Ln) bytes of Xn (tmp) to Cn to create En */ /* 5. Append the tail (BB - Ln) bytes of Xn to Cn to create En */
memcpy(s + bsize + lastn, tmp + lastn, bsize - lastn); memcpy(d + lastn, d + bsize + lastn, bsize - lastn);
/* 6. Decrypt En to create Pn-1 */ /* 6. Decrypt En to create Pn-1 */
memzero_explicit(iv, sizeof(iv));
sg_set_buf(&sgsrc[0], s + bsize, bsize); scatterwalk_map_and_copy(d, sg, 0, bsize + lastn, 1);
sg_set_buf(&sgdst[0], d, bsize); memzero_explicit(d, sizeof(d));
err = crypto_blkcipher_decrypt_iv(&lcldesc, sgdst, sgsrc, bsize);
/* XOR with previous block */ skcipher_request_set_callback(subreq, req->base.flags &
crypto_xor(d, desc->info, bsize); CRYPTO_TFM_REQ_MAY_BACKLOG,
cts_cbc_crypt_done, req);
scatterwalk_map_and_copy(d, dst, offset, nbytes, 1); skcipher_request_set_crypt(subreq, sg, sg, bsize, space);
return crypto_skcipher_decrypt(subreq);
memcpy(desc->info, s, bsize);
return err;
} }
static int crypto_cts_decrypt(struct blkcipher_desc *desc, static void crypto_cts_decrypt_done(struct crypto_async_request *areq, int err)
struct scatterlist *dst, struct scatterlist *src,
unsigned int nbytes)
{ {
struct crypto_cts_ctx *ctx = crypto_blkcipher_ctx(desc->tfm); struct skcipher_request *req = areq->data;
int bsize = crypto_blkcipher_blocksize(desc->tfm);
int tot_blocks = (nbytes + bsize - 1) / bsize;
int cbc_blocks = tot_blocks > 2 ? tot_blocks - 2 : 0;
struct blkcipher_desc lcldesc;
int err;
lcldesc.tfm = ctx->child; if (err)
lcldesc.info = desc->info; goto out;
lcldesc.flags = desc->flags;
if (tot_blocks == 1) { err = cts_cbc_decrypt(req);
err = crypto_blkcipher_decrypt_iv(&lcldesc, dst, src, bsize); if (err == -EINPROGRESS ||
} else if (nbytes <= bsize * 2) { (err == -EBUSY && req->base.flags & CRYPTO_TFM_REQ_MAY_BACKLOG))
err = cts_cbc_decrypt(ctx, desc, dst, src, 0, nbytes); return;
} else {
/* do normal function for tot_blocks - 2 */ out:
err = crypto_blkcipher_decrypt_iv(&lcldesc, dst, src, skcipher_request_complete(req, err);
cbc_blocks * bsize);
if (err == 0) {
/* do cts for final two blocks */
err = cts_cbc_decrypt(ctx, desc, dst, src,
cbc_blocks * bsize,
nbytes - (cbc_blocks * bsize));
}
}
return err;
} }
static int crypto_cts_init_tfm(struct crypto_tfm *tfm) static int crypto_cts_decrypt(struct skcipher_request *req)
{ {
struct crypto_instance *inst = (void *)tfm->__crt_alg; struct crypto_skcipher *tfm = crypto_skcipher_reqtfm(req);
struct crypto_spawn *spawn = crypto_instance_ctx(inst); struct crypto_cts_reqctx *rctx = skcipher_request_ctx(req);
struct crypto_cts_ctx *ctx = crypto_tfm_ctx(tfm); struct crypto_cts_ctx *ctx = crypto_skcipher_ctx(tfm);
struct crypto_blkcipher *cipher; struct skcipher_request *subreq = &rctx->subreq;
int bsize = crypto_skcipher_blocksize(tfm);
unsigned int nbytes = req->cryptlen;
int cbc_blocks = (nbytes + bsize - 1) / bsize - 1;
unsigned int offset;
u8 *space;
cipher = crypto_spawn_blkcipher(spawn); skcipher_request_set_tfm(subreq, ctx->child);
if (cbc_blocks <= 0) {
skcipher_request_set_callback(subreq, req->base.flags,
req->base.complete,
req->base.data);
skcipher_request_set_crypt(subreq, req->src, req->dst, nbytes,
req->iv);
return crypto_skcipher_decrypt(subreq);
}
skcipher_request_set_callback(subreq, req->base.flags,
crypto_cts_decrypt_done, req);
space = crypto_cts_reqctx_space(req);
offset = cbc_blocks * bsize;
rctx->offset = offset;
if (cbc_blocks <= 1)
memcpy(space, req->iv, bsize);
else
scatterwalk_map_and_copy(space, req->src, offset - 2 * bsize,
bsize, 0);
skcipher_request_set_crypt(subreq, req->src, req->dst,
offset, req->iv);
return crypto_skcipher_decrypt(subreq) ?:
cts_cbc_decrypt(req);
}
static int crypto_cts_init_tfm(struct crypto_skcipher *tfm)
{
struct skcipher_instance *inst = skcipher_alg_instance(tfm);
struct crypto_skcipher_spawn *spawn = skcipher_instance_ctx(inst);
struct crypto_cts_ctx *ctx = crypto_skcipher_ctx(tfm);
struct crypto_skcipher *cipher;
unsigned reqsize;
unsigned bsize;
unsigned align;
cipher = crypto_spawn_skcipher2(spawn);
if (IS_ERR(cipher)) if (IS_ERR(cipher))
return PTR_ERR(cipher); return PTR_ERR(cipher);
ctx->child = cipher; ctx->child = cipher;
align = crypto_skcipher_alignmask(tfm);
bsize = crypto_skcipher_blocksize(cipher);
reqsize = ALIGN(sizeof(struct crypto_cts_reqctx) +
crypto_skcipher_reqsize(cipher),
crypto_tfm_ctx_alignment()) +
(align & ~(crypto_tfm_ctx_alignment() - 1)) + bsize;
crypto_skcipher_set_reqsize(tfm, reqsize);
return 0; return 0;
} }
static void crypto_cts_exit_tfm(struct crypto_tfm *tfm) static void crypto_cts_exit_tfm(struct crypto_skcipher *tfm)
{ {
struct crypto_cts_ctx *ctx = crypto_tfm_ctx(tfm); struct crypto_cts_ctx *ctx = crypto_skcipher_ctx(tfm);
crypto_free_blkcipher(ctx->child);
crypto_free_skcipher(ctx->child);
} }
static struct crypto_instance *crypto_cts_alloc(struct rtattr **tb) static void crypto_cts_free(struct skcipher_instance *inst)
{ {
struct crypto_instance *inst; crypto_drop_skcipher(skcipher_instance_ctx(inst));
struct crypto_alg *alg; kfree(inst);
}
static int crypto_cts_create(struct crypto_template *tmpl, struct rtattr **tb)
{
struct crypto_skcipher_spawn *spawn;
struct skcipher_instance *inst;
struct crypto_attr_type *algt;
struct skcipher_alg *alg;
const char *cipher_name;
int err; int err;
err = crypto_check_attr_type(tb, CRYPTO_ALG_TYPE_BLKCIPHER); algt = crypto_get_attr_type(tb);
if (IS_ERR(algt))
return PTR_ERR(algt);
if ((algt->type ^ CRYPTO_ALG_TYPE_SKCIPHER) & algt->mask)
return -EINVAL;
cipher_name = crypto_attr_alg_name(tb[1]);
if (IS_ERR(cipher_name))
return PTR_ERR(cipher_name);
inst = kzalloc(sizeof(*inst) + sizeof(*spawn), GFP_KERNEL);
if (!inst)
return -ENOMEM;
spawn = skcipher_instance_ctx(inst);
crypto_set_skcipher_spawn(spawn, skcipher_crypto_instance(inst));
err = crypto_grab_skcipher2(spawn, cipher_name, 0,
crypto_requires_sync(algt->type,
algt->mask));
if (err) if (err)
return ERR_PTR(err); goto err_free_inst;
alg = crypto_attr_alg(tb[1], CRYPTO_ALG_TYPE_BLKCIPHER, alg = crypto_spawn_skcipher_alg(spawn);
CRYPTO_ALG_TYPE_MASK);
if (IS_ERR(alg))
return ERR_CAST(alg);
inst = ERR_PTR(-EINVAL); err = -EINVAL;
if (!is_power_of_2(alg->cra_blocksize)) if (crypto_skcipher_alg_ivsize(alg) != alg->base.cra_blocksize)
goto out_put_alg; goto err_drop_spawn;
if (strncmp(alg->cra_name, "cbc(", 4)) if (strncmp(alg->base.cra_name, "cbc(", 4))
goto out_put_alg; goto err_drop_spawn;
inst = crypto_alloc_instance("cts", alg); err = crypto_inst_setname(skcipher_crypto_instance(inst), "cts",
if (IS_ERR(inst)) &alg->base);
goto out_put_alg; if (err)
goto err_drop_spawn;
inst->alg.cra_flags = CRYPTO_ALG_TYPE_BLKCIPHER; inst->alg.base.cra_flags = alg->base.cra_flags & CRYPTO_ALG_ASYNC;
inst->alg.cra_priority = alg->cra_priority; inst->alg.base.cra_priority = alg->base.cra_priority;
inst->alg.cra_blocksize = alg->cra_blocksize; inst->alg.base.cra_blocksize = alg->base.cra_blocksize;
inst->alg.cra_alignmask = alg->cra_alignmask; inst->alg.base.cra_alignmask = alg->base.cra_alignmask;
inst->alg.cra_type = &crypto_blkcipher_type;
/* We access the data as u32s when xoring. */ /* We access the data as u32s when xoring. */
inst->alg.cra_alignmask |= __alignof__(u32) - 1; inst->alg.base.cra_alignmask |= __alignof__(u32) - 1;
inst->alg.cra_blkcipher.ivsize = alg->cra_blocksize; inst->alg.ivsize = alg->base.cra_blocksize;
inst->alg.cra_blkcipher.min_keysize = alg->cra_blkcipher.min_keysize; inst->alg.chunksize = crypto_skcipher_alg_chunksize(alg);
inst->alg.cra_blkcipher.max_keysize = alg->cra_blkcipher.max_keysize; inst->alg.min_keysize = crypto_skcipher_alg_min_keysize(alg);
inst->alg.max_keysize = crypto_skcipher_alg_max_keysize(alg);
inst->alg.cra_ctxsize = sizeof(struct crypto_cts_ctx); inst->alg.base.cra_ctxsize = sizeof(struct crypto_cts_ctx);
inst->alg.cra_init = crypto_cts_init_tfm; inst->alg.init = crypto_cts_init_tfm;
inst->alg.cra_exit = crypto_cts_exit_tfm; inst->alg.exit = crypto_cts_exit_tfm;
inst->alg.cra_blkcipher.setkey = crypto_cts_setkey; inst->alg.setkey = crypto_cts_setkey;
inst->alg.cra_blkcipher.encrypt = crypto_cts_encrypt; inst->alg.encrypt = crypto_cts_encrypt;
inst->alg.cra_blkcipher.decrypt = crypto_cts_decrypt; inst->alg.decrypt = crypto_cts_decrypt;
out_put_alg: inst->free = crypto_cts_free;
crypto_mod_put(alg);
return inst;
}
static void crypto_cts_free(struct crypto_instance *inst) err = skcipher_register_instance(tmpl, inst);
{ if (err)
crypto_drop_spawn(crypto_instance_ctx(inst)); goto err_drop_spawn;
out:
return err;
err_drop_spawn:
crypto_drop_skcipher(spawn);
err_free_inst:
kfree(inst); kfree(inst);
goto out;
} }
static struct crypto_template crypto_cts_tmpl = { static struct crypto_template crypto_cts_tmpl = {
.name = "cts", .name = "cts",
.alloc = crypto_cts_alloc, .create = crypto_cts_create,
.free = crypto_cts_free,
.module = THIS_MODULE, .module = THIS_MODULE,
}; };