1
0
mirror of git://git.gnupg.org/gnupg.git synced 2025-01-03 12:11:33 +01:00

gpg: Fix AEAD encryption for chunk sizes other than 64 KiB.

* g10/cipher-aead.c (do_flush): Init ERR.  Fix remaining chunklen
computation.
(do_free): Add dummy encryption.  Close the cipher handle.
* g10/decrypt-data.c (aead_underflow): Rewrite.
--

Until we have integrated test into the test suite extensive tests can
also be done with a script like this:

--8<---------------cut here---------------start------------->8---
#!/bin/sh

set -e
GPG="../g10/gpg --rfc4880bis --pinentry-mode=loopback"
GPG="$GPG --passphrase abc --batch"
MKTDATA="$HOME/b/gnupg-2.0/tools/mk-tdata"

for chunksize in 6 7 12 13 14 30; do
for count in $(seq 1 200) $(seq 8100 8200) \
             $(seq 16350 16400) $(seq 20000 20100); do
  if [ ! -f "testfile-$count" ]; then
    $MKTDATA $count >"testfile-$count"
  fi
  echo "testing chunk size 2^$chunksize with $count bytes"
  $GPG --force-aead --aead-algo ocb --s2k-mode 0 --cipher AES -v -z 0 \
      -c --chunk-size $chunksize \
       <"testfile-$count" >"testfile-$count.gpg" 2>/dev/null
  $GPG -vd <"testfile-$count.gpg" >"testfile-$count.out" 2>/dev/null
  if ! cmp "testfile-$count" "testfile-$count.out"; then
    echo "FAILED comparing count $count" >&2
    exit 1
  fi
done
done
echo All good
--8<---------------cut here---------------end--------------->8---

Signed-off-by: Werner Koch <wk@gnupg.org>
This commit is contained in:
Werner Koch 2018-01-24 13:45:05 +01:00
parent 83a15fa88e
commit ff1bdc23d9
No known key found for this signature in database
GPG Key ID: E3FDFF218E45B72B
2 changed files with 250 additions and 230 deletions

View File

@ -271,7 +271,7 @@ write_final_chunk (cipher_filter_context_t *cfx, iobuf_t a)
static gpg_error_t static gpg_error_t
do_flush (cipher_filter_context_t *cfx, iobuf_t a, byte *buf, size_t size) do_flush (cipher_filter_context_t *cfx, iobuf_t a, byte *buf, size_t size)
{ {
gpg_error_t err; gpg_error_t err = 0;
int newchunk = 0; int newchunk = 0;
size_t n; size_t n;
@ -285,9 +285,9 @@ do_flush (cipher_filter_context_t *cfx, iobuf_t a, byte *buf, size_t size)
else else
n = cfx->bufsize - cfx->buflen; n = cfx->bufsize - cfx->buflen;
if (cfx->chunklen + n >= cfx->chunksize) if (cfx->chunklen + cfx->buflen + n >= cfx->chunksize)
{ {
size_t n1 = cfx->chunksize - cfx->chunklen; size_t n1 = cfx->chunksize - (cfx->chunklen + cfx->buflen);
newchunk = 1; newchunk = 1;
if (DBG_FILTER) if (DBG_FILTER)
log_debug ("chunksize %ju reached;" log_debug ("chunksize %ju reached;"
@ -305,8 +305,8 @@ do_flush (cipher_filter_context_t *cfx, iobuf_t a, byte *buf, size_t size)
if (cfx->buflen == cfx->bufsize || newchunk) if (cfx->buflen == cfx->bufsize || newchunk)
{ {
if (DBG_FILTER) if (DBG_FILTER)
log_debug ("encrypting: buflen=%zu %s %p\n", log_debug ("encrypting: buflen=%zu %s n=%zu\n",
cfx->buflen, newchunk?"(newchunk)":"", cfx->cipher_hd); cfx->buflen, newchunk?"(newchunk)":"", n);
if (newchunk) if (newchunk)
gcry_cipher_final (cfx->cipher_hd); gcry_cipher_final (cfx->cipher_hd);
if (!DBG_FILTER) if (!DBG_FILTER)
@ -372,6 +372,9 @@ do_free (cipher_filter_context_t *cfx, iobuf_t a)
{ {
gpg_error_t err = 0; gpg_error_t err = 0;
if (DBG_FILTER)
log_debug ("do_free: buflen=%zu\n", cfx->buflen);
/* FIXME: Check what happens if we just wrote the last chunk and no /* FIXME: Check what happens if we just wrote the last chunk and no
* more bytes were to encrypt. We should then not call finalize and * more bytes were to encrypt. We should then not call finalize and
* write the auth tag again, right? May this at all happen? */ * write the auth tag again, right? May this at all happen? */
@ -394,6 +397,8 @@ do_free (cipher_filter_context_t *cfx, iobuf_t a)
cfx->chunklen += cfx->buflen; cfx->chunklen += cfx->buflen;
cfx->total += cfx->buflen; cfx->total += cfx->buflen;
} }
else /* Dummy encryption. */
gcry_cipher_encrypt (cfx->cipher_hd, cfx->buffer, 0, NULL, 0);
/* Get and write the authentication tag. */ /* Get and write the authentication tag. */
if (DBG_FILTER) if (DBG_FILTER)
@ -411,8 +416,8 @@ do_free (cipher_filter_context_t *cfx, iobuf_t a)
leave: leave:
xfree (cfx->buffer); xfree (cfx->buffer);
cfx->buffer = NULL; cfx->buffer = NULL;
/* gcry_cipher_close (cfx->cipher_hd); */ gcry_cipher_close (cfx->cipher_hd);
/* cfx->cipher_hd = NULL; */ cfx->cipher_hd = NULL;
return err; return err;
} }

View File

@ -56,8 +56,8 @@ struct decode_filter_context_s
/* The start IV for AEAD encryption. */ /* The start IV for AEAD encryption. */
byte startiv[16]; byte startiv[16];
/* The holdback buffer and its used length. For AEAD we need at /* The holdback buffer and its used length. For AEAD we need 32+1
* least 32+1 byte for MDC 22 bytes are required. */ * bytes but we use 48 byte. For MDC we need 22 bytes. */
char holdback[48]; char holdback[48];
unsigned int holdbacklen; unsigned int holdbacklen;
@ -536,262 +536,277 @@ decrypt_data (ctrl_t ctrl, void *procctx, PKT_encrypted *ed, DEK *dek)
static gpg_error_t static gpg_error_t
aead_underflow (decode_filter_ctx_t dfx, iobuf_t a, byte *buf, size_t *ret_len) aead_underflow (decode_filter_ctx_t dfx, iobuf_t a, byte *buf, size_t *ret_len)
{ {
const size_t size = *ret_len; /* The initial length of BUF. */ const size_t size = *ret_len; /* The allocated size of BUF. */
gpg_error_t err; gpg_error_t err;
size_t n; /* Finally the number of decrypted bytes in BUF. */ size_t totallen = 0; /* The number of bytes to return on success or EOF. */
size_t off = 0; /* The offset into the buffer. */
size_t len; /* The current number of bytes in BUF+OFF. */
int c; int c;
log_assert (size > 64); /* Our code requires at least this size. */ log_assert (size > 48); /* Our code requires at least this size. */
/* Get at least 32 bytes and put it ahead in the buffer. */ /* Copy the rest from the last call of this function into BUF. */
if (dfx->partial) len = dfx->holdbacklen;
{ dfx->holdbacklen = 0;
for (n=32; n < 64; n++) memcpy (buf, dfx->holdback, len);
{
if ((c = iobuf_get (a)) == -1)
break;
buf[n] = c;
}
}
else
{
for (n=32; n < 64 && dfx->length; n++, dfx->length--)
{
if ((c = iobuf_get (a)) == -1)
break; /* Premature EOF. */
buf[n] = c;
}
}
if (n == 64)
{
/* We got 32 bytes from A which are good for the last chunk's
* auth tag and the final chunk's auth tag. On the first time
* we don't have anything in the holdback buffer and thus we move
* those 32 bytes to the start of the buffer. All further calls
* will copy the 32 bytes from the holdback buffer to the start of the
* buffer. */
if (!dfx->holdbacklen)
{
memcpy (buf, buf+32, 32);
n = 32; /* Continue at this position. */
}
else
{
memcpy (buf, dfx->holdback, 32);
}
/* Now fill up the provided buffer. */
if (dfx->partial)
{
for (; n < size; n++ )
{
if ((c = iobuf_get (a)) == -1)
{
dfx->eof_seen = 1; /* Normal EOF. */
break;
}
buf[n] = c;
}
}
else
{
for (; n < size && dfx->length; n++, dfx->length--)
{
c = iobuf_get (a);
if (c == -1)
{
dfx->eof_seen = 3; /* Premature EOF. */
break;
}
buf[n] = c;
}
if (!dfx->length)
dfx->eof_seen = 1; /* Normal EOF. */
}
/* Move the trailing 32 bytes back to the holdback buffer. We
* got at least 64 bytes and thus a memmove is not needed. */
n -= 32;
memcpy (dfx->holdback, buf+n, 32);
dfx->holdbacklen = 32;
}
else if (!dfx->holdbacklen)
{
/* EOF seen but empty holdback buffer. This means that we did
* not read enough for the two auth tags. */
n -= 32;
memcpy (buf, buf+32, n );
dfx->eof_seen = 2; /* EOF with incomplete tag. */
}
else
{
/* EOF seen (i.e. read less than 32 bytes). */
memcpy (buf, dfx->holdback, 32);
n -= 32;
memcpy (dfx->holdback, buf+n, 32);
dfx->eof_seen = 1; /* Normal EOF. */
}
if (DBG_FILTER) if (DBG_FILTER)
log_debug ("decrypt: chunklen=%ju total=%ju size=%zu n=%zu%s\n", log_debug ("aead_underflow: size=%zu len=%zu%s\n",
(uintmax_t)dfx->chunklen, (uintmax_t)dfx->total, size, n, size, len, dfx->eof_seen? " eof":"");
dfx->eof_seen? " eof":"");
/* Now decrypt the buffer. */ /* Read and fill up BUF. We need to watchout for an EOF so that we
if (n && dfx->eof_seen > 1) * can detect the last chunk which is commonly shorter than the
* chunksize. After the last data byte from the last chunk 32 more
* bytes are expected for the last chunk's tag and the following
* final chunk's tag. To detect the EOF we need to read at least
* one further byte; however we try to ready 16 extra bytes to avoid
* singel byte reads in some lower layers. The outcome is that we
* have up to 48 extra extra octets which we will later put into the
* holdback buffer for the next invocation (which handles the EOF
* case). */
if (dfx->partial)
{ {
err = gpg_error (GPG_ERR_TRUNCATED); for (; len < size; len++ )
} {
else if (!n) if ((c = iobuf_get (a)) == -1)
{ {
log_assert (dfx->eof_seen); dfx->eof_seen = 1; /* Normal EOF. */
err = gpg_error (GPG_ERR_EOF); break;
}
buf[len] = c;
}
} }
else else
{ {
size_t off = 0; for (; len < size && dfx->length; len++, dfx->length--)
if (dfx->chunklen + n >= dfx->chunksize)
{ {
size_t n0 = dfx->chunksize - dfx->chunklen; c = iobuf_get (a);
if (c == -1)
if (DBG_FILTER)
log_debug ("chunksize will be reached: n0=%zu\n", n0);
gcry_cipher_final (dfx->cipher_hd);
err = gcry_cipher_decrypt (dfx->cipher_hd, buf, n0, NULL, 0);
if (err)
{ {
log_error ("gcry_cipher_decrypt failed (1): %s\n", dfx->eof_seen = 3; /* Premature EOF. */
gpg_strerror (err)); break;
goto leave;
} }
/*log_printhex (buf, n, "buf:");*/ buf[len] = c;
dfx->chunklen += n0;
dfx->total += n0;
off = n0;
n -= n0;
if (DBG_FILTER)
log_debug ("bytes left: %zu off=%zu\n", n, off);
log_assert (n >= 16);
log_assert (dfx->holdbacklen);
if (DBG_CRYPTO)
log_printhex (buf+off, 16, "tag:");
err = gcry_cipher_checktag (dfx->cipher_hd, buf + off, 16);
if (err)
{
if (DBG_FILTER)
log_debug ("gcry_cipher_checktag failed (1): %s\n",
gpg_strerror (err));
/* Return Bad Signature like we do with MDC encryption. */
if (gpg_err_code (err) == GPG_ERR_CHECKSUM)
err = gpg_error (GPG_ERR_BAD_SIGNATURE);
goto leave;
}
/* Remove that tag from the output. */
memmove (buf + off, buf + off + 16, n - 16);
n -= 16;
/* Prepare a new chunk. */
dfx->chunklen = 0;
dfx->chunkindex++;
err = aead_set_nonce (dfx);
if (err)
goto leave;
err = aead_set_ad (dfx, 0);
if (err)
goto leave;
} }
if (!dfx->length)
dfx->eof_seen = 1; /* Normal EOF. */
}
if (dfx->eof_seen) if (len < 32)
{ {
/* This is the last block of the last chunk. Its length may /* Not enough data for the last two tags. */
* not be a multiple of the block length. We expect that it err = gpg_error (GPG_ERR_TRUNCATED);
* is followed by two authtags. The first being the one goto leave;
* from the current chunk and the second form the final }
* chunk encrypting the empty string. Note that for the if (dfx->eof_seen)
* other blocks we assume a multiple of the block length {
* which is only true because the filter is called with /* If have seen an EOF we copy only the last two auth tags into
* large 2^n sized buffers. There is no assert because * the holdback buffer. */
* gcry_cipher_decrypt would detect such an error. */ dfx->holdbacklen = 32;
gcry_cipher_final (dfx->cipher_hd); memcpy (dfx->holdback, buf+len-32, 32);
/* log_printhex (buf+off, n, "buf+off:"); */ len -= 32;
} }
err = gcry_cipher_decrypt (dfx->cipher_hd, buf + off, n, NULL, 0); else
{
/* If have not seen an EOF we copy the entire extra 48 bytes
* into the holdback buffer for processing at the next call of
* this function. */
dfx->holdbacklen = len > 48? 48 : len;
memcpy (dfx->holdback, buf+len-dfx->holdbacklen, dfx->holdbacklen);
len -= dfx->holdbacklen;
}
/* log_printhex (dfx->holdback, dfx->holdbacklen, "holdback:"); */
/* Decrypt the buffer. This requires a loop because a chunk may end
* within the buffer. */
if (DBG_FILTER)
log_debug ("decrypt loop: chunklen=%ju total=%ju size=%zu len=%zu%s\n",
(uintmax_t)dfx->chunklen, (uintmax_t)dfx->total, size, len,
dfx->eof_seen? " eof":"");
while (len && dfx->chunklen + len >= dfx->chunksize)
{
size_t n = dfx->chunksize - dfx->chunklen;
byte tagbuf[16];
if (DBG_FILTER)
log_debug ("chunksize will be reached: n=%zu\n", n);
/* log_printhex (buf, n, "ciph:"); */
gcry_cipher_final (dfx->cipher_hd);
err = gcry_cipher_decrypt (dfx->cipher_hd, buf+off, n, NULL, 0);
if (err) if (err)
{ {
log_error ("gcry_cipher_decrypt failed (2): %s\n", log_error ("gcry_cipher_decrypt failed (1): %s\n",
gpg_strerror (err)); gpg_strerror (err));
goto leave; goto leave;
} }
/* log_printhex (buf, n, "plai:"); */
totallen += n;
dfx->chunklen += n; dfx->chunklen += n;
dfx->total += n; dfx->total += n;
off += n;
len -= n;
if (dfx->eof_seen) if (DBG_FILTER)
log_debug ("bytes left: %zu at off=%zu\n", len, off);
/* Check the tag. */
if (len < 16)
{ {
log_printhex (buf+off, n, "buf+off:"); /* The tag is not entirely in the buffer. Read the rest of
if (DBG_FILTER) * the tag from the holdback buffer. The shift the holdback
log_debug ("eof seen: chunklen=%ju total=%ju off=%zu n=%zu\n", * buffer and fill it up again. */
(uintmax_t)dfx->chunklen, (uintmax_t)dfx->total, off, n); memcpy (tagbuf, buf+off, len);
memcpy (tagbuf + len, dfx->holdback, 16 - len);
dfx->holdbacklen -= 16-len;
memmove (dfx->holdback, dfx->holdback + (16-len), dfx->holdbacklen);
log_assert (dfx->holdbacklen); len = dfx->holdbacklen;
err = gcry_cipher_checktag (dfx->cipher_hd, dfx->holdback, 16); if (dfx->partial)
if (err)
{ {
log_printhex (dfx->holdback, 16, "tag:"); for (; len < 48; len++ )
log_error ("gcry_cipher_checktag failed (2): %s\n", {
gpg_strerror (err)); if ((c = iobuf_get (a)) == -1)
/* Return Bad Signature like we do with MDC encryption. */ {
if (gpg_err_code (err) == GPG_ERR_CHECKSUM) dfx->eof_seen = 1; /* Normal EOF. */
err = gpg_error (GPG_ERR_BAD_SIGNATURE); break;
}
dfx->holdback[len] = c;
}
}
else
{
for (; len < 48 && dfx->length; len++, dfx->length--)
{
c = iobuf_get (a);
if (c == -1)
{
dfx->eof_seen = 3; /* Premature EOF. */
break;
}
dfx->holdback[len] = c;
}
if (!dfx->length)
dfx->eof_seen = 1; /* Normal EOF. */
}
if (len < 32)
{
/* Not enough data for the last two tags. */
err = gpg_error (GPG_ERR_TRUNCATED);
goto leave; goto leave;
} }
dfx->holdbacklen = len;
/* Check the final chunk. */ /* log_printhex (dfx->holdback, dfx->holdbacklen, "holdback:"); */
dfx->chunkindex++; len = 0;
err = aead_set_nonce (dfx);
if (err)
goto leave;
err = aead_set_ad (dfx, 1);
if (err)
goto leave;
gcry_cipher_final (dfx->cipher_hd);
/* decrypt an empty string. */
err = gcry_cipher_decrypt (dfx->cipher_hd, buf, 0, NULL, 0);
if (err)
{
log_error ("gcry_cipher_decrypt failed (final): %s\n",
gpg_strerror (err));
goto leave;
}
err = gcry_cipher_checktag (dfx->cipher_hd, dfx->holdback+16, 16);
if (err)
{
if (DBG_FILTER)
log_debug ("gcry_cipher_checktag failed (final): %s\n",
gpg_strerror (err));
/* Return Bad Signature like we do with MDC encryption. */
if (gpg_err_code (err) == GPG_ERR_CHECKSUM)
err = gpg_error (GPG_ERR_BAD_SIGNATURE);
goto leave;
}
n += off;
if (DBG_FILTER)
log_debug ("eof seen: returning %zu\n", n);
/* log_printhex (buf, n, "buf:"); */
} }
else else /* We already have the full tag. */
n += off; {
memcpy (tagbuf, buf+off, 16);
/* Remove that tag from the output. */
memmove (buf + off, buf + off + 16, len - 16);
len -= 16;
}
if (DBG_CRYPTO)
log_printhex (tagbuf, 16, "tag:");
err = gcry_cipher_checktag (dfx->cipher_hd, tagbuf, 16);
if (err)
{
if (DBG_FILTER)
log_debug ("gcry_cipher_checktag failed (1): %s\n",
gpg_strerror (err));
goto leave;
}
/* Prepare a new chunk. */
dfx->chunklen = 0;
dfx->chunkindex++;
err = aead_set_nonce (dfx);
if (err)
goto leave;
err = aead_set_ad (dfx, 0);
if (err)
goto leave;
continue;
}
if (dfx->eof_seen)
{
/* This is the last block of the last chunk. Its length may
* not be a multiple of the block length. */
gcry_cipher_final (dfx->cipher_hd);
}
err = gcry_cipher_decrypt (dfx->cipher_hd, buf + off, len, NULL, 0);
if (err)
{
log_error ("gcry_cipher_decrypt failed (2): %s\n", gpg_strerror (err));
goto leave;
}
totallen += len;
dfx->chunklen += len;
dfx->total += len;
if (dfx->eof_seen)
{
if (DBG_FILTER)
log_debug ("eof seen: holdback buffer has the tags.\n");
log_assert (dfx->holdbacklen >= 32);
if (DBG_FILTER)
log_printhex (dfx->holdback, 16, "tag:");
err = gcry_cipher_checktag (dfx->cipher_hd, dfx->holdback, 16);
if (err)
{
log_error ("gcry_cipher_checktag failed (2): %s\n",
gpg_strerror (err));
goto leave;
}
/* Check the final chunk. */
dfx->chunkindex++;
err = aead_set_nonce (dfx);
if (err)
goto leave;
err = aead_set_ad (dfx, 1);
if (err)
goto leave;
gcry_cipher_final (dfx->cipher_hd);
/* decrypt an empty string. */
err = gcry_cipher_decrypt (dfx->cipher_hd, dfx->holdback, 0, NULL, 0);
if (err)
{
log_error ("gcry_cipher_decrypt failed (final): %s\n",
gpg_strerror (err));
goto leave;
}
/* log_printhex (dfx->holdback+16, 16, "tag:"); */
err = gcry_cipher_checktag (dfx->cipher_hd, dfx->holdback+16, 16);
if (err)
{
if (DBG_FILTER)
log_debug ("gcry_cipher_checktag failed (final): %s\n",
gpg_strerror (err));
goto leave;
}
err = gpg_error (GPG_ERR_EOF);
} }
leave: leave:
/* In case of a real error we better wipe out the buffer than to if (DBG_FILTER)
* keep partly encrypted data. */ log_debug ("aead_underflow: returning %zu (%s)\n",
totallen, gpg_strerror (err));
/* In case of an auth error we map the error code to the same as
* used by the MDC decryption. */
if (gpg_err_code (err) == GPG_ERR_CHECKSUM)
err = gpg_error (GPG_ERR_BAD_SIGNATURE);
/* In case of an error we better wipe out the buffer than to convey
* partly decrypted data. */
if (err && gpg_err_code (err) != GPG_ERR_EOF) if (err && gpg_err_code (err) != GPG_ERR_EOF)
memset (buf, 0, size); memset (buf, 0, size);
*ret_len = n;
*ret_len = totallen;
return err; return err;
} }