Commit 0de686f7 authored by Harald Freudenberger's avatar Harald Freudenberger Committed by Greg Kroah-Hartman

s390/crypto: fix gcm-aes-s390 selftest failures

commit bef9f0ba upstream.

The current kernel uses improved crypto selftests. These
tests showed that the current implementation of gcm-aes-s390
is not able to deal with chunks of output buffers which are
not a multiple of 16 bytes. This patch introduces a rework
of the gcm aes s390 scatter walk handling which now is able
to handle any input and output scatter list chunk sizes
correctly.

Code has been verified by the crypto selftests, the tcrypt
kernel module and additional tests ran via the af_alg interface.

Cc: <stable@vger.kernel.org>
Reported-by: default avatarJulian Wiedmann <jwi@linux.ibm.com>
Reviewed-by: default avatarPatrick Steuer <steuer@linux.ibm.com>
Signed-off-by: default avatarHarald Freudenberger <freude@linux.ibm.com>
Signed-off-by: default avatarHeiko Carstens <heiko.carstens@de.ibm.com>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
parent 273ec0ff
...@@ -826,7 +826,7 @@ static int gcm_aes_setauthsize(struct crypto_aead *tfm, unsigned int authsize) ...@@ -826,7 +826,7 @@ static int gcm_aes_setauthsize(struct crypto_aead *tfm, unsigned int authsize)
return 0; return 0;
} }
static void gcm_sg_walk_start(struct gcm_sg_walk *gw, struct scatterlist *sg, static void gcm_walk_start(struct gcm_sg_walk *gw, struct scatterlist *sg,
unsigned int len) unsigned int len)
{ {
memset(gw, 0, sizeof(*gw)); memset(gw, 0, sizeof(*gw));
...@@ -834,11 +834,37 @@ static void gcm_sg_walk_start(struct gcm_sg_walk *gw, struct scatterlist *sg, ...@@ -834,11 +834,37 @@ static void gcm_sg_walk_start(struct gcm_sg_walk *gw, struct scatterlist *sg,
scatterwalk_start(&gw->walk, sg); scatterwalk_start(&gw->walk, sg);
} }
static int gcm_sg_walk_go(struct gcm_sg_walk *gw, unsigned int minbytesneeded) static inline unsigned int _gcm_sg_clamp_and_map(struct gcm_sg_walk *gw)
{
struct scatterlist *nextsg;
gw->walk_bytes = scatterwalk_clamp(&gw->walk, gw->walk_bytes_remain);
while (!gw->walk_bytes) {
nextsg = sg_next(gw->walk.sg);
if (!nextsg)
return 0;
scatterwalk_start(&gw->walk, nextsg);
gw->walk_bytes = scatterwalk_clamp(&gw->walk,
gw->walk_bytes_remain);
}
gw->walk_ptr = scatterwalk_map(&gw->walk);
return gw->walk_bytes;
}
static inline void _gcm_sg_unmap_and_advance(struct gcm_sg_walk *gw,
unsigned int nbytes)
{
gw->walk_bytes_remain -= nbytes;
scatterwalk_unmap(&gw->walk);
scatterwalk_advance(&gw->walk, nbytes);
scatterwalk_done(&gw->walk, 0, gw->walk_bytes_remain);
gw->walk_ptr = NULL;
}
static int gcm_in_walk_go(struct gcm_sg_walk *gw, unsigned int minbytesneeded)
{ {
int n; int n;
/* minbytesneeded <= AES_BLOCK_SIZE */
if (gw->buf_bytes && gw->buf_bytes >= minbytesneeded) { if (gw->buf_bytes && gw->buf_bytes >= minbytesneeded) {
gw->ptr = gw->buf; gw->ptr = gw->buf;
gw->nbytes = gw->buf_bytes; gw->nbytes = gw->buf_bytes;
...@@ -851,13 +877,11 @@ static int gcm_sg_walk_go(struct gcm_sg_walk *gw, unsigned int minbytesneeded) ...@@ -851,13 +877,11 @@ static int gcm_sg_walk_go(struct gcm_sg_walk *gw, unsigned int minbytesneeded)
goto out; goto out;
} }
gw->walk_bytes = scatterwalk_clamp(&gw->walk, gw->walk_bytes_remain); if (!_gcm_sg_clamp_and_map(gw)) {
if (!gw->walk_bytes) { gw->ptr = NULL;
scatterwalk_start(&gw->walk, sg_next(gw->walk.sg)); gw->nbytes = 0;
gw->walk_bytes = scatterwalk_clamp(&gw->walk, goto out;
gw->walk_bytes_remain);
} }
gw->walk_ptr = scatterwalk_map(&gw->walk);
if (!gw->buf_bytes && gw->walk_bytes >= minbytesneeded) { if (!gw->buf_bytes && gw->walk_bytes >= minbytesneeded) {
gw->ptr = gw->walk_ptr; gw->ptr = gw->walk_ptr;
...@@ -869,51 +893,90 @@ static int gcm_sg_walk_go(struct gcm_sg_walk *gw, unsigned int minbytesneeded) ...@@ -869,51 +893,90 @@ static int gcm_sg_walk_go(struct gcm_sg_walk *gw, unsigned int minbytesneeded)
n = min(gw->walk_bytes, AES_BLOCK_SIZE - gw->buf_bytes); n = min(gw->walk_bytes, AES_BLOCK_SIZE - gw->buf_bytes);
memcpy(gw->buf + gw->buf_bytes, gw->walk_ptr, n); memcpy(gw->buf + gw->buf_bytes, gw->walk_ptr, n);
gw->buf_bytes += n; gw->buf_bytes += n;
gw->walk_bytes_remain -= n; _gcm_sg_unmap_and_advance(gw, n);
scatterwalk_unmap(&gw->walk);
scatterwalk_advance(&gw->walk, n);
scatterwalk_done(&gw->walk, 0, gw->walk_bytes_remain);
if (gw->buf_bytes >= minbytesneeded) { if (gw->buf_bytes >= minbytesneeded) {
gw->ptr = gw->buf; gw->ptr = gw->buf;
gw->nbytes = gw->buf_bytes; gw->nbytes = gw->buf_bytes;
goto out; goto out;
} }
if (!_gcm_sg_clamp_and_map(gw)) {
gw->walk_bytes = scatterwalk_clamp(&gw->walk, gw->ptr = NULL;
gw->walk_bytes_remain); gw->nbytes = 0;
if (!gw->walk_bytes) { goto out;
scatterwalk_start(&gw->walk, sg_next(gw->walk.sg));
gw->walk_bytes = scatterwalk_clamp(&gw->walk,
gw->walk_bytes_remain);
} }
gw->walk_ptr = scatterwalk_map(&gw->walk);
} }
out: out:
return gw->nbytes; return gw->nbytes;
} }
static void gcm_sg_walk_done(struct gcm_sg_walk *gw, unsigned int bytesdone) static int gcm_out_walk_go(struct gcm_sg_walk *gw, unsigned int minbytesneeded)
{ {
int n; if (gw->walk_bytes_remain == 0) {
gw->ptr = NULL;
gw->nbytes = 0;
goto out;
}
if (!_gcm_sg_clamp_and_map(gw)) {
gw->ptr = NULL;
gw->nbytes = 0;
goto out;
}
if (gw->walk_bytes >= minbytesneeded) {
gw->ptr = gw->walk_ptr;
gw->nbytes = gw->walk_bytes;
goto out;
}
scatterwalk_unmap(&gw->walk);
gw->walk_ptr = NULL;
gw->ptr = gw->buf;
gw->nbytes = sizeof(gw->buf);
out:
return gw->nbytes;
}
static int gcm_in_walk_done(struct gcm_sg_walk *gw, unsigned int bytesdone)
{
if (gw->ptr == NULL) if (gw->ptr == NULL)
return; return 0;
if (gw->ptr == gw->buf) { if (gw->ptr == gw->buf) {
n = gw->buf_bytes - bytesdone; int n = gw->buf_bytes - bytesdone;
if (n > 0) { if (n > 0) {
memmove(gw->buf, gw->buf + bytesdone, n); memmove(gw->buf, gw->buf + bytesdone, n);
gw->buf_bytes -= n; gw->buf_bytes = n;
} else } else
gw->buf_bytes = 0; gw->buf_bytes = 0;
} else { } else
gw->walk_bytes_remain -= bytesdone; _gcm_sg_unmap_and_advance(gw, bytesdone);
scatterwalk_unmap(&gw->walk);
scatterwalk_advance(&gw->walk, bytesdone); return bytesdone;
scatterwalk_done(&gw->walk, 0, gw->walk_bytes_remain); }
static int gcm_out_walk_done(struct gcm_sg_walk *gw, unsigned int bytesdone)
{
int i, n;
if (gw->ptr == NULL)
return 0;
if (gw->ptr == gw->buf) {
for (i = 0; i < bytesdone; i += n) {
if (!_gcm_sg_clamp_and_map(gw))
return i;
n = min(gw->walk_bytes, bytesdone - i);
memcpy(gw->walk_ptr, gw->buf + i, n);
_gcm_sg_unmap_and_advance(gw, n);
} }
} else
_gcm_sg_unmap_and_advance(gw, bytesdone);
return bytesdone;
} }
static int gcm_aes_crypt(struct aead_request *req, unsigned int flags) static int gcm_aes_crypt(struct aead_request *req, unsigned int flags)
...@@ -926,7 +989,7 @@ static int gcm_aes_crypt(struct aead_request *req, unsigned int flags) ...@@ -926,7 +989,7 @@ static int gcm_aes_crypt(struct aead_request *req, unsigned int flags)
unsigned int pclen = req->cryptlen; unsigned int pclen = req->cryptlen;
int ret = 0; int ret = 0;
unsigned int len, in_bytes, out_bytes, unsigned int n, len, in_bytes, out_bytes,
min_bytes, bytes, aad_bytes, pc_bytes; min_bytes, bytes, aad_bytes, pc_bytes;
struct gcm_sg_walk gw_in, gw_out; struct gcm_sg_walk gw_in, gw_out;
u8 tag[GHASH_DIGEST_SIZE]; u8 tag[GHASH_DIGEST_SIZE];
...@@ -963,14 +1026,14 @@ static int gcm_aes_crypt(struct aead_request *req, unsigned int flags) ...@@ -963,14 +1026,14 @@ static int gcm_aes_crypt(struct aead_request *req, unsigned int flags)
*(u32 *)(param.j0 + ivsize) = 1; *(u32 *)(param.j0 + ivsize) = 1;
memcpy(param.k, ctx->key, ctx->key_len); memcpy(param.k, ctx->key, ctx->key_len);
gcm_sg_walk_start(&gw_in, req->src, len); gcm_walk_start(&gw_in, req->src, len);
gcm_sg_walk_start(&gw_out, req->dst, len); gcm_walk_start(&gw_out, req->dst, len);
do { do {
min_bytes = min_t(unsigned int, min_bytes = min_t(unsigned int,
aadlen > 0 ? aadlen : pclen, AES_BLOCK_SIZE); aadlen > 0 ? aadlen : pclen, AES_BLOCK_SIZE);
in_bytes = gcm_sg_walk_go(&gw_in, min_bytes); in_bytes = gcm_in_walk_go(&gw_in, min_bytes);
out_bytes = gcm_sg_walk_go(&gw_out, min_bytes); out_bytes = gcm_out_walk_go(&gw_out, min_bytes);
bytes = min(in_bytes, out_bytes); bytes = min(in_bytes, out_bytes);
if (aadlen + pclen <= bytes) { if (aadlen + pclen <= bytes) {
...@@ -997,8 +1060,11 @@ static int gcm_aes_crypt(struct aead_request *req, unsigned int flags) ...@@ -997,8 +1060,11 @@ static int gcm_aes_crypt(struct aead_request *req, unsigned int flags)
gw_in.ptr + aad_bytes, pc_bytes, gw_in.ptr + aad_bytes, pc_bytes,
gw_in.ptr, aad_bytes); gw_in.ptr, aad_bytes);
gcm_sg_walk_done(&gw_in, aad_bytes + pc_bytes); n = aad_bytes + pc_bytes;
gcm_sg_walk_done(&gw_out, aad_bytes + pc_bytes); if (gcm_in_walk_done(&gw_in, n) != n)
return -ENOMEM;
if (gcm_out_walk_done(&gw_out, n) != n)
return -ENOMEM;
aadlen -= aad_bytes; aadlen -= aad_bytes;
pclen -= pc_bytes; pclen -= pc_bytes;
} while (aadlen + pclen > 0); } while (aadlen + pclen > 0);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment